Compare commits
40 Commits
3bc0825e0b
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d5ce0e2e2b | ||
|
|
deb6afff3e | ||
|
|
b4818349be | ||
|
|
ff9d0f41d1 | ||
|
|
7371c3703d | ||
|
|
851f76ca16 | ||
|
|
a29896a790 | ||
|
|
3f513f6c12 | ||
|
|
8f39dfbc1e | ||
|
|
e1243f603e | ||
|
|
80d90f317d | ||
|
|
12021eb269 | ||
|
|
308fee6c27 | ||
|
|
756f77730f | ||
|
|
c05fa00007 | ||
|
|
ce90dac833 | ||
|
|
c9f9a1ca0e | ||
|
|
9eccb9f6fa | ||
|
|
8dd0e1678c | ||
|
|
3fc53534e2 | ||
|
|
9e42e7ed6f | ||
|
|
19f89ecd70 | ||
|
|
9031f81908 | ||
|
|
4caf055c53 | ||
|
|
822185e138 | ||
|
|
41aa385644 | ||
|
|
865c208821 | ||
|
|
670019361c | ||
|
|
8d4ff3400e | ||
|
|
bafaf05e83 | ||
|
|
b26dbb79e1 | ||
|
|
283299ff20 | ||
|
|
412102b9b1 | ||
|
|
a4531acd69 | ||
|
|
d487afad73 | ||
|
|
c31a1e254c | ||
|
|
4a2afc68bf | ||
|
|
f05fe5b221 | ||
|
|
074b6e7c8a | ||
|
|
5a964a1a8d |
@@ -19,3 +19,9 @@ AUTOPASS_WORKER_USE_TEST_CREDS=false
|
||||
# --- RAR ---
|
||||
# test | prod
|
||||
AUTOPASS_RAR_ENV=test
|
||||
|
||||
# --- Embeddings (sugestie mapare, Stratul 2 PRD 5.14) ---
|
||||
# false = dezactivat (default; /mapari instant, sugestii din GOLD/SILVER + fuzzy).
|
||||
# true = sugestii semantice. Prima cerere /mapari lazy-load-eaza modelul fastembed/ONNX
|
||||
# (~230MB pe disc) sincron -> hang la prima cerere. Doar API-ul il incarca.
|
||||
AUTOPASS_EMBEDDINGS_ENABLED=false
|
||||
|
||||
@@ -67,7 +67,7 @@ Flux: validare (`validation.py`) → mapare operatie→cod (`mapping.py`) → en
|
||||
- **Idempotenta = hash de continut canonic** server-side (`idempotency.py`), pentru ca RAR accepta duplicate si nu are nr. comanda. `build_key` normalizeaza INTOTDEAUNA `account_id` la `account_or_default` (None == 1) INAINTE de hash — altfel acelasi rand logic primeste chei diferite pe canalele API vs import (OV-2). `canonicalize_row` normeaza VIN/nr/odometru (strip ".0" din coercion Excel) inainte de validare si de cheie.
|
||||
- **`FINALIZATA` e terminal la RAR** — fara anulare/corectie prin API. De aceea reconcilierea anti-duplicat: pe eroare **ambigua** (timeout / TransportError / 502/503/504 / 429 / 408) sau rand `sending` orfan, worker-ul cauta in finalizate (match pe vin+dataPrestatie+odometruFinal) si marcheaza `sent` fara a re-trimite (`reconcile.py`). **EXCEPTIE: un RAR 500 cu mesaj** (`RarError.rar_message`, ex. `ORA-12899`) e un esec DEFINITIV (RAR a raspuns „am esuat", nu o pierdere de raspuns) → worker-ul NU reconciliaza si NU reincearca, marcheaza `error` cu mesajul RAR (`RAR_EROARE_SERVER`). Altfel ar marca fals `sent` pe un record PARTIAL pe care RAR (ne-tranzactional) il lasa la esec.
|
||||
- **Creds RAR per cont**: durabile in `accounts.rar_creds_enc` (canal web, fallback re-login) SAU efemere in `submissions.rar_creds_enc` (canal API, sterse dupa primul login reusit). Worker incearca submission-ul intai, apoi fallback la cont. Purjarea sterge DOAR `submissions.rar_creds_enc`, NU `accounts.rar_creds_enc`.
|
||||
- **Auth API-key** (`auth.py`): identifica CONTUL ROAAUTO, separat de credentialele RAR. Stocam doar SHA-256 al cheii. Enforcement prin `AUTOPASS_REQUIRE_API_KEY`: `false` (dev) → fara cheie merge pe cont id=1, cheie invalida → 401; `true` (prod) → cheie obligatorie pe `/v1/*` protejat. POST-urile + rutele de import sunt account-scoped; GET-urile de listare sunt momentan globale + neprotejate (de remediat — vezi ROADMAP).
|
||||
- **Auth API-key** (`auth.py`): identifica CONTUL ROAAUTO, separat de credentialele RAR. Stocam doar SHA-256 al cheii. Enforcement prin `AUTOPASS_REQUIRE_API_KEY`: `false` (dev) → fara cheie merge pe cont id=1, cheie invalida → 401; `true` (prod) → cheie obligatorie pe `/v1/*` protejat. POST-urile + rutele de import sunt account-scoped; GET-urile de listare sunt si ele account-scoped (5.15/US-011: fragmentele `_fragments/submissions|trimitere|mapari|status|jurnal|nomenclator|trimiteri-versiune` sub `require_login` + scope, 404-before-leak pe id strain; `GET /v1/prezentari(/{id})`/`/v1/mapari`/`/v1/audit/export` filtrate pe cont). `GET /v1/nomenclator` ramane public intentionat (coduri RAR publice, fara PII).
|
||||
- **Mapare coloane retinuta per `(account_id, signature_coloane)`** (`column_mappings`): la urmatorul fisier cu aceleasi coloane, pentru acelasi cont, maparea se reaplica automat. Un cont poate avea mai multe formate memorate simultan.
|
||||
- **Mapare operatie→cod**: prestatie poate veni cu `cod_prestatie` (cod RAR direct) sau `cod_op_service` (cod intern) + `denumire`. Nerezolvat → submission `needs_mapping` (nu se trimite), apare in editorul web cu sugestie fuzzy; la salvarea maparii se re-rezolva automat submission-urile blocate.
|
||||
- **`cod_prestatie` e VALIDAT fata de nomenclator la ingestie** (`resolve_prestatii(..., valid_codes)`): un cod direct NECUNOSCUT in nomenclator NU se mai trimite raw — e promovat la `cod_op_service` (denumire=cod) si tratat ca operatie de mapat. Motiv (confirmat live 2026-06-23): RAR accepta NUMAI coduri din nomenclator (coloana `COD_PRESTATIE` max 5 car.); un cod necunoscut da **HTTP 500** (`ORA-12899`), iar RAR **NU e tranzactional** → lasa un record PARTIAL `FINALIZATA` (terminal) chiar pe esec, pe care reconcilierea worker-ului l-ar marca fals `sent`. Comportamentul la cod necunoscut/nemapat: `on_unmapped_error` (camp boolean top-level pe `POST /v1/prezentari` + `/valideaza`) = `false` (intra in editor, `needs_mapping`) sau `true` (respinge fara enqueue → `submission_id=null` + `erori`). Default = `accounts.on_unmapped_error_default` (implicit `false`/`0`); precedenta cerere > cont > `false`.
|
||||
|
||||
341
DESIGN.md
Normal file
341
DESIGN.md
Normal file
@@ -0,0 +1,341 @@
|
||||
# DESIGN.md — Sistem de design AutoPass (by ROMFAST)
|
||||
|
||||
> Sursa de adevar pentru identitatea vizuala a dashboard-ului. Implementarea concreta sta in
|
||||
> `app/web/templates/base.html` (variabile CSS `:root` + `[data-theme="light"]`). Acest fisier
|
||||
> spune *ce* si *de ce*; base.html spune *cum*.
|
||||
|
||||
## Lucrul de retinut
|
||||
|
||||
> „Software serios pentru o obligatie legala serioasa — dar parte din familia ROMFAST/ROA, nu un
|
||||
> tool anonim." Operatorul de service trebuie sa simta ca declara la stat printr-un instrument de
|
||||
> incredere, cu identitatea producatorului (ROMFAST) prezenta discret, nu griul generic de azi.
|
||||
|
||||
## Context produs
|
||||
|
||||
Gateway web care declara prezentari de service-auto la RAR AUTOPASS (L.142/2023). Utilizatori:
|
||||
operatori de service-auto si integratori ROAAUTO. Face parte din familia **ROA — Romfast
|
||||
Applications** (ERP romanesc, modul Service Auto). Referinta de brand: **romfast.ro** — alb curat,
|
||||
accent albastru azur, pill-uri rotunjite, comutator de tema, logo rosu+albastru.
|
||||
|
||||
## Decizie cromatica
|
||||
|
||||
Accentul functional = **albastrul ROMFAST** (acelasi cu „FAST" din logo si cu accentul de pe
|
||||
romfast.ro), nu albastrul generic SaaS de pana acum. Rosul apare DOAR in wordmark-ul „ROM" — nu ca
|
||||
accent de UI, fiindca rosul e rezervat starilor de eroare. Un singur accent, restul neutre, ca
|
||||
sistemul sa ramana discret.
|
||||
|
||||
### Paleta — Dark (default)
|
||||
|
||||
```
|
||||
--bg: #0f1218 fundal aplicatie
|
||||
--card: #181c24 suprafete (carduri, modal, inputuri pe fundal)
|
||||
--card2: #0f1218 fundal input slim / carduri-contor (= --bg, nivelul cel mai adanc)
|
||||
--ink: #e6e9ef text principal
|
||||
--muted: #8b93a7 text secundar (label-uri, coduri, „by")
|
||||
--line: #262b36 borduri, separatoare
|
||||
--line2: #1f2530 separator subtire lista slim (mai subtil decat --line)
|
||||
--accent:#2E74D6 azur ROMFAST — butoane primare, pill activ, linkuri, focus
|
||||
--ok: #2FBF8F sent / succes
|
||||
--warn: #E0A93B sending / atentie / Lipsa cod
|
||||
--err: #E05D5D error / needs_data / Date incomplete
|
||||
```
|
||||
|
||||
### Paleta — Light (`[data-theme="light"]`)
|
||||
|
||||
```
|
||||
--bg: #f5f7fa fundal (alb-rece ca romfast.ro)
|
||||
--card: #ffffff suprafete
|
||||
--card2: #f5f7fa fundal input slim / carduri-contor (= --bg)
|
||||
--ink: #1a1d24 text principal
|
||||
--muted: #5c6473 text secundar
|
||||
--line: #e2e5ea borduri
|
||||
--line2: #eaedf2 separator subtire lista slim (mai subtil decat --line)
|
||||
--accent:#1F66C9 azur, varianta mai inchisa pentru contrast AA pe alb
|
||||
--ok: #15803d verde AA pe alb
|
||||
--warn: #b45309 chihlimbar AA pe alb
|
||||
--err: #dc2626 rosu AA pe alb
|
||||
```
|
||||
|
||||
### Paleta — Petrol (`[data-theme="petrol"]`, tema selectabila)
|
||||
|
||||
Tema intunecata alternativa, cu accent petrol-teal (directia initiala aleasa, pastrata ca optiune).
|
||||
Aceleasi neutre-calde inchise; doar accentul difera de azur.
|
||||
|
||||
```
|
||||
--bg: #0e1416 fundal petrol-inchis
|
||||
--card: #161e20 suprafete
|
||||
--card2: #0e1416 fundal input/contor (= --bg)
|
||||
--ink: #e6e9ef text principal
|
||||
--muted: #8b93a7 text secundar
|
||||
--line: #232c2e borduri
|
||||
--line2: #1c2426 separator subtire (intre --bg si --line)
|
||||
--accent:#0E7C7B teal petrol — butoane, pill activ, linkuri, focus
|
||||
--ok: #2FBF8F sent
|
||||
--warn: #E0A93B atentie
|
||||
--err: #E05D5D eroare
|
||||
```
|
||||
|
||||
### Paleta — Grafit (`[data-theme="grafit"]`, tema selectabila — adaugata PRD 5.15)
|
||||
|
||||
Similara cu dark, accent azur deschis (preluat din landing, `--infot`). Distinta de dark la cererea
|
||||
userului (D2). Mapare landing->app: `--text->--ink`, `--sub->--muted`, `--okt->--ok`,
|
||||
`--errt->--err`, `--infot->--accent`.
|
||||
|
||||
```
|
||||
--bg: #0f1218 fundal (identic cu dark)
|
||||
--card: #181c24 suprafete
|
||||
--card2: #0f1218 fundal input/contor (= --bg)
|
||||
--ink: #e6e9ef text principal
|
||||
--muted: #8b93a7 text secundar
|
||||
--line: #262b36 borduri
|
||||
--line2: #1f2530 separator subtire
|
||||
--accent:#6ea2ec azur deschis (landing --infot) — linkuri, focus, pill activ
|
||||
--ok: #2FBF8F sent / succes
|
||||
--warn: #E0A93B atentie
|
||||
--err: #E05D5D eroare
|
||||
```
|
||||
|
||||
### Paleta — Cobalt (`[data-theme="cobalt"]`, tema selectabila — adaugata PRD 5.15)
|
||||
|
||||
Fundal bleumarin adanc, accent albastru viu. Atmosfera tehnica/corporatista rece.
|
||||
|
||||
```
|
||||
--bg: #080d1c fundal bleumarin adanc
|
||||
--card: #111a33 suprafete
|
||||
--card2: #0b1226 fundal input/contor
|
||||
--ink: #e9ecfb text principal (usor albastrat)
|
||||
--muted: #8a93b8 text secundar
|
||||
--line: #1d2747 borduri
|
||||
--line2: #161f3a separator subtire
|
||||
--accent:#8aa0ff albastru viu (landing --infot)
|
||||
--ok: #2fd0a6 sent / succes (teal mai saturat)
|
||||
--warn: #E0A93B atentie
|
||||
--err: #f06a7a eroare (roz saturat pe bleumarin)
|
||||
```
|
||||
|
||||
### Paleta — Cupru (`[data-theme="cupru"]`, tema selectabila — adaugata PRD 5.15)
|
||||
|
||||
Fundal cald ciocolata, accent chihlimbar. Atmosfera artizanala/calda.
|
||||
|
||||
```
|
||||
--bg: #15110b fundal maro inchis-cald
|
||||
--card: #211a12 suprafete
|
||||
--card2: #15110b fundal input/contor (= --bg)
|
||||
--ink: #efe6d6 text principal (crem cald)
|
||||
--muted: #a89a85 text secundar
|
||||
--line: #36291c borduri
|
||||
--line2: #281e14 separator subtire
|
||||
--accent:#dfa45c chihlimbar cald (landing --infot)
|
||||
--ok: #67b98c sent / succes (verde muted-cald)
|
||||
--warn: #c97d2e atentie (chihlimbar mai inchis)
|
||||
--err: #e2685a eroare (coral pe maro)
|
||||
```
|
||||
|
||||
### Paleta — Hartie (`[data-theme="hartie"]`, tema selectabila — adaugata PRD 5.15)
|
||||
|
||||
Fundal crem cald (hartie vintage), accent albastru clasic. Similara cu light, distinta la cererea
|
||||
userului. Ambele teme luminoase (hartie + light) respecta contrast AA.
|
||||
|
||||
```
|
||||
--bg: #f3efe6 fundal crem cald
|
||||
--card: #fffdf7 suprafete (crem-alb)
|
||||
--card2: #f3efe6 fundal input/contor (= --bg)
|
||||
--ink: #1e1a13 text principal (maro-inchis, AA pe crem)
|
||||
--muted: #6a6052 text secundar
|
||||
--line: #e2dccc borduri
|
||||
--line2: #ece6d9 separator subtire (mai deschis decat line)
|
||||
--accent:#1F5FBF albastru clasic (landing --infot = --accent) — 6.5:1 pe --bg, AA
|
||||
--ok: #1c7d5d sent / succes (verde AA pe crem)
|
||||
--warn: #b45309 atentie (chihlimbar AA pe crem)
|
||||
--err: #bd463c eroare (rosu AA pe crem)
|
||||
```
|
||||
|
||||
### Tokeni noi adaugati la PRD 5.15 (in toate cele 7 teme)
|
||||
|
||||
```
|
||||
--card2 fundal input slim si carduri-contor (US-001/002); pe dark = --bg (cel mai adanc nivel)
|
||||
--line2 separator subtire intre randuri lista slim (US-001/002); mai subtil decat --line
|
||||
```
|
||||
|
||||
### Culori de brand (doar wordmark, NU variabile de UI)
|
||||
|
||||
```
|
||||
ROM: #D1342F rosu logo
|
||||
FAST: #2E74D6 albastru logo (= accentul de UI in dark)
|
||||
```
|
||||
|
||||
Contrast: textul principal pe fundal ramane AA in ambele teme; accentul pe alb foloseste varianta
|
||||
mai inchisa (`#1F66C9`) ca text/linkul sa treaca AA.
|
||||
|
||||
## Tipografie
|
||||
|
||||
- **UI / titluri**: **IBM Plex Sans** — sans-serif cu caracter ingineresc, open-source, potrivit
|
||||
pentru „software serios", parte din limbajul vizual tehnic. Fallback: `system-ui, sans-serif`.
|
||||
- **Coduri / monospace**: **IBM Plex Mono** — pentru coduri RAR (REV2), VIN, numar inmatriculare,
|
||||
detalii tehnice. Inlocuieste `ui-monospace/Menlo` actual cu o familie coerenta cu UI-ul.
|
||||
- **Incarcare**: self-host `woff2` (subset latin + latin-ext pentru diacritice romanesti) in
|
||||
`app/web/static/fonts/`, `font-display: swap`. Fara CDN extern (gateway intern, fara dependente
|
||||
de retea la runtime). Pana la self-host, fallback la stiva de sistem nu strica layout-ul.
|
||||
|
||||
## Header & branding
|
||||
|
||||
- Titlul „Gateway RAR AUTOPASS" **centrat** pe header.
|
||||
- Sub titlu, mic: **logo-ul ROMFAST** (`/static/romfast_logo.png`, ~28px inaltime). Decizie user
|
||||
(2026-06-25, US-012b): se foloseste PNG-ul real al logo-ului (ROM rosu + FAST albastru, fundal
|
||||
transparent — lizibil pe light/dark/petrol), NU wordmark-ul text. Wordmark-ul text (`by ROM FAST`
|
||||
cu `ROM #D1342F` / `FAST #2E74D6`) ramane documentat ca alternativa, dar livrabila finala
|
||||
foloseste imaginea.
|
||||
- Controalele (comutator tema, versiune, hamburger ☰) raman la **dreapta**, fara a strica
|
||||
centrarea optica a titlului (ex. grila 3 coloane: stanga goala/echilibru, centru titlu, dreapta
|
||||
controale).
|
||||
- Responsiv: pe mobil, wordmark-ul ramane sub titlu; controalele nu se suprapun (degrada elegant,
|
||||
eventual titlu mai mic).
|
||||
|
||||
## Selector de tema
|
||||
|
||||
Inlocuieste comutatorul binar soare/luna cu un **buton ciclic** (pattern ca demoanaf.ro): un
|
||||
singur buton care roteste la fiecare click prin setul de teme, cu iconita + tooltip/`aria-label`
|
||||
care arata tema curenta („Tema: Light" etc.).
|
||||
|
||||
Ordinea ciclului (PRD 5.15 — teme aditive D2):
|
||||
**Light → Dark → Petrol → Grafit → Cobalt → Cupru → Hartie → Auto → (inapoi la Light)**.
|
||||
|
||||
- `Light` → `data-theme="light"` (azur pe alb) — ☀
|
||||
- `Dark` → `data-theme="dark"` (azur pe inchis, comportamentul implicit actual) — ☾
|
||||
- `Petrol` → `data-theme="petrol"` (teal pe petrol-inchis) — ◐
|
||||
- `Grafit` → `data-theme="grafit"` (azur deschis pe negru-grafit, similar dark) — ◑
|
||||
- `Cobalt` → `data-theme="cobalt"` (albastru viu pe bleumarin adanc) — ◆
|
||||
- `Cupru` → `data-theme="cupru"` (chihlimbar pe maro cald) — ◇
|
||||
- `Hartie` → `data-theme="hartie"` (albastru clasic pe crem cald, similar light) — ○
|
||||
- `Auto` → urmeaza `prefers-color-scheme`; rezolva la `light` (OS light) sau `dark` (OS dark). — ◉
|
||||
|
||||
Persistenta: preferinta explicita (inclusiv „Auto") in `localStorage`, doar la click. Scriptul
|
||||
anti-FOUC din `<head>` cunoaste toate cele 7+1 stari; valori vechi (light/dark/petrol) raman
|
||||
valide fara migrare fortata; valoare lipsa/necunoscuta → auto (fallback sigur, fara blink).
|
||||
|
||||
Implementare DRY (E2 PRD 5.15): configuratia temelor traieste intr-o singura structura JS
|
||||
`var THEMES = [...]` (sursa de adevar), din care se DERIVA `CYCLE`/`VALID`/`ICONS`/`LABELS`/`NEXT`.
|
||||
Adaugarea unei teme noi = O singura intrare in `THEMES`.
|
||||
|
||||
Default la prima vizita = Auto (OS-aware), ca inainte.
|
||||
|
||||
## Componente — note de aplicare
|
||||
|
||||
- **Pill-uri de stare/filtru**: rotunjite (`border-radius:99px`), ca badge-ul „ROA" de pe
|
||||
romfast.ro. Pill activ = fundal accent discret (`color-mix(in srgb, var(--accent) ...)`), text
|
||||
pe accent. Categoriile de problema isi pastreaza registrul: Date incomplete/Eroare = `--err`,
|
||||
Lipsa cod = `--warn`.
|
||||
- **Butoane primare**: fundal `--accent`, text alb (neschimbat ca structura, doar culoarea noua).
|
||||
- **Linkuri / sugestii**: `--accent`.
|
||||
- **Focus**: `outline:2px solid var(--accent)` (deja folosit pe randuri).
|
||||
- **Suprafete de stare** (banner, flash, eroare-3n): raman pe `color-mix` peste `--err/--warn/--ok`,
|
||||
deci se adapteaza automat la noua paleta si la light/dark.
|
||||
|
||||
## Componente slim (PRD 5.15 US-002)
|
||||
|
||||
Adaugate in `base.html` (sectiunea `SENTINEL-COMPONENTE-SLIM`). Toate culorile exclusiv prin
|
||||
`var(--token)` — zero hex hardcodat. Consumate de US-003 (dashboard), US-004 (lista), US-007 (formular).
|
||||
|
||||
### `.contor-card`
|
||||
|
||||
Card cifra-contor compact: fundal `--card2`, bordura `--line`, `border-radius:8px`, padding 10-12px.
|
||||
|
||||
```html
|
||||
<div class="contor-card">
|
||||
<div class="contor-cifra s-ok">847</div> <!-- variante de culoare prin .s-ok/.s-err/.s-queued -->
|
||||
<div class="contor-label">Trimise (total)</div>
|
||||
<div class="contor-sub">luna 124 · azi 9</div> <!-- optional: sub-linie mono -->
|
||||
</div>
|
||||
```
|
||||
|
||||
Sub-elemente:
|
||||
- `.contor-cifra` — `font-size:22px; font-weight:700`; culoare prin `.s-*` existente
|
||||
- `.contor-label` — `font-size:11px; color:var(--muted)`
|
||||
- `.contor-sub` — IBM Plex Mono, `font-size:10px; color:var(--muted)`
|
||||
|
||||
### `.lista-trimiteri-slim` + `.trimitere-slim`
|
||||
|
||||
Lista compacta cu separator `--line2`. Randul este clickabil (rol button), tinta `min-height:44px`.
|
||||
|
||||
```html
|
||||
<ul class="lista-trimiteri-slim">
|
||||
<li class="trimitere-slim" role="button" tabindex="0">
|
||||
<div>
|
||||
<div class="slim-vin">WBA8E9...K7F2</div>
|
||||
<div class="slim-meta">Inspectie tehnica · 09:42</div>
|
||||
</div>
|
||||
<span class="pill s-sent">Trimis</span>
|
||||
</li>
|
||||
</ul>
|
||||
```
|
||||
|
||||
Sub-elemente:
|
||||
- `.slim-vin` — IBM Plex Mono, `font-size:13px; font-weight:500; color:var(--ink)`
|
||||
- `.slim-meta` — `font-size:11px; color:var(--muted)` (operatie + ora)
|
||||
|
||||
### `.camp-slim` + macro `camp(slim=True)`
|
||||
|
||||
Varianta compacta de camp formular: label 11px muted deasupra, input `height:30px`, fundal `--card2`.
|
||||
Integrata in macro-ul `camp` din `_macros.html` prin flagul `slim=False` (default — randarea
|
||||
actuala ramane neschimbata).
|
||||
|
||||
```jinja2
|
||||
{{ camp('vin', 'VIN (serie sasiu)', vin, slim=True) }}
|
||||
```
|
||||
|
||||
Pentru campuri mono (VIN, odometru, nr. inmatriculare): adauga clasa `camp-mono` pe input
|
||||
(via `style=""` sau atribut `class=""` direct — macro-ul nu il pune automat, consumatorul decide).
|
||||
|
||||
### `.chips` + `.chip` + `.chip-del`
|
||||
|
||||
Prestatii multi-select: container `.chips` (fundal `--card2`), item `.chip` (accent 18%, IBM Plex
|
||||
Mono 11px), buton de stergere `.chip-del` (accesibil cu `aria-label`).
|
||||
|
||||
```html
|
||||
<div class="chips" role="group" aria-label="Prestatii selectate">
|
||||
<span class="chip">
|
||||
<button class="chip-del" aria-label="Sterge codul REV2" type="button">×</button>
|
||||
REV2
|
||||
</span>
|
||||
<span class="chip chip-warn"> <!-- varianta warn pentru R-ODO/I-ODO -->
|
||||
<button class="chip-del" aria-label="Sterge codul R-ODO" type="button">×</button>
|
||||
R-ODO
|
||||
</span>
|
||||
</div>
|
||||
```
|
||||
|
||||
Clase aditionale:
|
||||
- `.chip-warn` — fundal `--warn` 22% (pentru coduri R-ODO/I-ODO care cer odometruInitial)
|
||||
|
||||
### `.add-code` + `.op-row` (picker E4)
|
||||
|
||||
Buton dashed pentru adaugare cod (`.add-code`) si randul operatie<->cod (`.op-row`, `.op-row-name`,
|
||||
`.op-row-warn`). Folosite de picker-ul E4 din US-007 (formular editare).
|
||||
|
||||
```html
|
||||
<div class="op-row">
|
||||
<span class="op-row-name">REVIZIE PERIODICA</span>
|
||||
<span class="chip">REV2 <button class="chip-del" ...>×</button></span>
|
||||
<button class="add-code" type="button">+ alt cod</button>
|
||||
</div>
|
||||
<div class="op-row op-row-warn"> <!-- bordura warn: lipsa cod -->
|
||||
<span class="op-row-name">SCHIMBARE PLACUTE FRANA</span>
|
||||
<button class="add-code" type="button">alege cod RAR</button>
|
||||
</div>
|
||||
```
|
||||
|
||||
## Ce NU schimbam
|
||||
|
||||
- Mecanismul light/dark existent (anti-FOUC, persistenta `localStorage`, comutator) — il pastram,
|
||||
doar reimprospatam variabilele.
|
||||
- Nu introducem rosu ca accent de UI (conflict cu eroare).
|
||||
- ~~Nu folosim PNG-ul logo cu efect 3D in interfata (wordmark redat ca text).~~ REVIZUIT
|
||||
(decizie user 2026-06-25): logo-ul PNG real e folosit in header (US-012b). Fundal transparent +
|
||||
culori proprii il fac lizibil pe toate temele; nu aplicam filtre.
|
||||
- Nu adaugam un al doilea accent — sistemul ramane monocrom-accent + neutre.
|
||||
|
||||
## Legatura cu implementarea (PRD 5.10)
|
||||
|
||||
US-012 (header „by ROMFAST" + titlu centrat) si US-013 (paleta) din
|
||||
`docs/prd/prd-5.10-ux-filtre-pill-paginare-mapari-meniu.md` implementeaza acest sistem. Valorile de
|
||||
mai sus sunt sursa pentru variabilele din `base.html`.
|
||||
@@ -3,10 +3,17 @@
|
||||
FROM python:3.12-slim
|
||||
|
||||
ENV PYTHONUNBUFFERED=1 \
|
||||
PYTHONDONTWRITEBYTECODE=1
|
||||
PYTHONDONTWRITEBYTECODE=1 \
|
||||
# Fus orar RO: SQLite 'localtime' (bucketare contoare azi/luna, E7) depinde de TZ.
|
||||
# tzdata ofera baza de fusuri; TZ alege Europe/Bucharest (DST-aware, UTC+2/+3).
|
||||
TZ=Europe/Bucharest
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# tzdata = necesar pentru ca 'localtime' din SQLite sa rezolve Europe/Bucharest.
|
||||
RUN apt-get update && apt-get install -y --no-install-recommends tzdata \
|
||||
&& rm -rf /var/lib/apt/lists/*
|
||||
|
||||
COPY requirements.txt .
|
||||
RUN pip install --no-cache-dir -r requirements.txt
|
||||
|
||||
|
||||
78
TODOS.md
Normal file
78
TODOS.md
Normal file
@@ -0,0 +1,78 @@
|
||||
# TODOS
|
||||
|
||||
Elemente deferate din review-uri. Negrupte de un PRD curent; de promovat cand devin prioritare.
|
||||
|
||||
## Din PRD 5.12 (2026-06-26)
|
||||
|
||||
- [ ] **Mai multi utilizatori per firma (flux de invitatie / alaturare la cont)** — azi CUI e unic, deci
|
||||
al doilea email care vrea pe aceeasi firma e respins la signup (nu exista flux de „alatura-te firmei").
|
||||
`users` permite tehnic mai multe loginuri per `account_id`, dar nu exista UI. Daca apare nevoia reala
|
||||
(mai multe persoane dintr-o firma), construieste: admin-ul firmei invita un email SAU al doilea cere
|
||||
acces si admin-ul aproba; membership pe `account_id`. Decizie user (2026-06-26): in 5.12 ramane
|
||||
**1 firma = 1 cont = 1 login** + mesaj prietenos la CUI duplicat (US-001); acest flux = livrabila separata.
|
||||
|
||||
## Din /autoplan PRD 5.11 (2026-06-26)
|
||||
|
||||
- [ ] **E2E smoke de first-run ca poarta de release** — codifica scriptul de dogfooding
|
||||
(import -> mapcoloane -> preview -> commit -> lista apare + contoare) ca test E2E care
|
||||
trebuie sa treaca inainte de orice release. Motiv: cele 8 bug-uri din 5.11 sunt toate
|
||||
first-run friction nedogfooded end-to-end; fara o poarta, reapar ca 8 tichete noi.
|
||||
(CEO F2, severitate high.)
|
||||
|
||||
- [ ] **Control compensator optional pe auto-trimitere unattended** — utilizatorul a ales
|
||||
(2026-06-26) scoaterea completa a hold-ului auto_send. Risc rezidual acceptat: o regula
|
||||
text gresita poate auto-trimite FINALIZATA (terminal, fara undo) pe randuri pe canalul API /
|
||||
remapare inline (fara gate de preview). Daca apar integratori reali, evalueaza un throttle
|
||||
„primele N auto-trimiteri pe o regula text noua cer confirmare" sau un kill-switch per cont.
|
||||
(CEO F5/F6, severitate critical ca risc, dar pre-launch exposure ~zero acum.)
|
||||
|
||||
## Din /autoplan PRD 5.13 (2026-06-27)
|
||||
|
||||
- [ ] **Filtre de data 2x2 pe mobil** — Azi/7zile/30zile/Custom stivuiesc full-width (4 randuri)
|
||||
pe mobil; grid 2x2 ar fi mai compact. Imbunatatire viitoare. (Design, low.)
|
||||
- [ ] **Sprite `<use href="#...">` pentru iconitele Lucide** — `act_btn` randeaza SVG inline pe
|
||||
fiecare rand (bloat DOM pe toate viewporturile, ascuns pe desktop). Optimizare deferata; inline
|
||||
acum (P5 simplu > optim prematur). (Eng §1, medium.)
|
||||
- [ ] **"Eroare/Eroare" la nivel routes.py/labels.py** — guard-ul de template (pill-only cand
|
||||
eticheta==stare) acopera cazul vizibil; curatarea logicii de continut ramane debt. (Design §2.)
|
||||
- [ ] **Validare premisa "utilizare mobil reala"** — inainte de orice extindere responsive viitoare,
|
||||
confirma device-mix-ul (analytics/cerere user). Daca ~95% desktop, nu mai investi in cardificare
|
||||
mobil. (CEO F1, high — premisa nedovedita acum.)
|
||||
|
||||
## Din /plan-ceo-review PRD 5.15 (2026-06-28)
|
||||
|
||||
- [ ] **Validare premisa "utilizare mobil reala" (reluare F1 din 5.13)** — partea slim/compact a lui
|
||||
5.15 presupune utilizare reala pe mobil. Inainte de orice rafinare responsive viitoare, confirma
|
||||
device-mix-ul (analytics / cerere user). Daca ~95% desktop, nu mai investi in cardificare mobil.
|
||||
(CEO, high — premisa nedovedita.)
|
||||
|
||||
- [ ] **Deduparea/etichetarea temelor grafit~dark si hartie~light** — 5.15 adauga 4 teme peste cele 3
|
||||
existente (7 + Auto). grafit e ~ identic cu dark, hartie ~ identic cu light. Daca selectorul devine
|
||||
confuz sau matricea de test apasa, dedupica sau eticheteaza-le clar. (CEO, low — simplificare optionala.)
|
||||
|
||||
- [ ] **US-009/US-010 ca PRD separat daca propagarea design e urgenta** — salvarea mapare-din-chip si
|
||||
bulk-fix sunt adiacente FUNCTIONALE (acceptate via SELECTIVE EXPANSION), dincolo de obiectivul pur de
|
||||
propagare design. Daca vrei sa livrezi designul rapid, pot fi scoase intr-un PRD propriu. (CEO, low.)
|
||||
|
||||
## Din raport comparatie mockup 5.16 (2026-06-29)
|
||||
|
||||
> Restul task-urilor din `docs/raport-comparatie-mockup-5.16.md` au fost livrate (T-1..T-9).
|
||||
> Cele de mai jos raman explicit in coada la cererea userului.
|
||||
|
||||
- [ ] **Stare de eroare HTMX la incarcarea listei (D-4)** — cand `/_fragments/submissions`
|
||||
da 500 sau pica reteaua, `#submissions-wrap` ramane blocat pe spinner ("se incarca…") fara
|
||||
mesaj. De adaugat un partial de eroare / `hx-on::response-error` cu "nu s-a putut incarca,
|
||||
reincearca". Robustete pre-existenta (nu introdusa de 5.16), impact functional real —
|
||||
**candidatul cu cea mai mare valoare** din lista. (Design D-4, medium.)
|
||||
|
||||
- [ ] **Retokenizare px completa in template-uri** — `_submissions.html` / `_preview_*` folosesc
|
||||
literali `font-size:13px/12px/11px` in loc de token-urile `--fs-*`. 5.16 a corectat doar
|
||||
instanta sub-12px (incalca pragul PRD). Restul ramane debt: schimbarea in masa (13px→`--fs-sm`
|
||||
=13.5px) misca layout-ul, deci necesita o baza de regresie vizuala inainte. (Eng, bounded —
|
||||
amanat ca scope creep fara baza AC.)
|
||||
|
||||
- [ ] **Diacritice in textul vizibil pentru user** — mockup-urile folosesc diacritice complete
|
||||
("Observații", "Salvează", "Adaugă"); aplicatia le omite in majoritatea label-urilor. Fontul
|
||||
le randeaza corect (US-001 confirmat). De aplicat pe label-uri/butoane/titluri, pastrand
|
||||
cod/comentariile fara diacritice. Decizie initiala (poarta de gust T3): nu se aplica acum —
|
||||
reintrodus in coada la cererea userului (2026-06-29) ca finisaj viitor. (Transversal, low.)
|
||||
146
app/accounts.py
146
app/accounts.py
@@ -15,34 +15,69 @@ inca fluxul de trimitere. (Addendum A2.)
|
||||
from __future__ import annotations
|
||||
|
||||
import sqlite3
|
||||
from datetime import datetime, timedelta, timezone
|
||||
|
||||
|
||||
def _norm_cui(cui: str | None) -> str | None:
|
||||
"""trim + upper; sir gol -> None (tratat ca „fara CUI")."""
|
||||
"""trim + upper; sir gol -> ValueError daca e string gol, None daca e None."""
|
||||
if cui is None:
|
||||
return None
|
||||
cui = cui.strip().upper()
|
||||
return cui or None
|
||||
if cui == "":
|
||||
raise ValueError("CUI gol (un CUI trebuie sa fie un sir nevid)")
|
||||
return cui
|
||||
|
||||
|
||||
def _norm_email(email: str | None) -> str | None:
|
||||
"""trim + lower; sir gol -> ValueError daca e string gol, None daca e None."""
|
||||
if email is None:
|
||||
return None
|
||||
email = email.strip().lower()
|
||||
if email == "":
|
||||
raise ValueError("email gol (un email trebuie sa fie un sir nevid)")
|
||||
return email
|
||||
|
||||
|
||||
def create_account(
|
||||
conn: sqlite3.Connection, name: str, cui: str | None = None, active: bool = True
|
||||
conn: sqlite3.Connection,
|
||||
name: str,
|
||||
cui: str | None = None,
|
||||
email: str | None = None,
|
||||
active: bool = True,
|
||||
requested_plan: str | None = None,
|
||||
consent_at: str | None = None,
|
||||
) -> int:
|
||||
"""Insereaza un cont si intoarce id-ul nou (AUTOINCREMENT, deci >=2 — nu atinge default id=1).
|
||||
|
||||
`name` gol/whitespace -> ValueError. `cui` se normalizeaza (trim+upper); un CUI
|
||||
deja folosit -> ValueError cu cauza+fix. Unicitatea e impusa de indexul partial
|
||||
`name` gol/whitespace -> ValueError. `cui` se normalizeaza (trim+upper); sir gol -> ValueError.
|
||||
`email` se normalizeaza (trim+lower); sir gol -> ValueError.
|
||||
Un CUI deja folosit -> ValueError cu cauza+fix. Unicitatea e impusa de indexul partial
|
||||
`ux_accounts_cui` (nu de un check separat), deci e sigura la concurenta.
|
||||
|
||||
`requested_plan`: planul CERUT la signup (separat de `tier`). NU acorda drepturi — `tier`
|
||||
ramane mereu 'free' la creare; planul cerut e doar o intentie pentru integrarea platilor.
|
||||
Valoare invalida (nu e in VALID_TIERS) -> ignorata (stocata NULL), nu arunca.
|
||||
`consent_at`: marca temporala consimtamant Termeni+GDPR (proba); None = fara flux consimtamant.
|
||||
"""
|
||||
name = (name or "").strip()
|
||||
if not name:
|
||||
raise ValueError("name gol (un cont are nevoie de nume)")
|
||||
cui = _norm_cui(cui)
|
||||
email = _norm_email(email)
|
||||
# Planul cerut: pastram doar valori valide; orice altceva -> NULL (defensiv).
|
||||
req_plan = requested_plan if requested_plan in VALID_TIERS else None
|
||||
try:
|
||||
# Trial Pro automat la creare (PRD 5.17 US-001): tier='free' + trial_until=now+30z.
|
||||
trial_until = (
|
||||
(datetime.now(timezone.utc) + timedelta(days=30)).strftime("%Y-%m-%d %H:%M:%S")
|
||||
)
|
||||
# Invariant (5.5): active=1 <=> status='active'; cont creat inactiv = 'pending'.
|
||||
cur = conn.execute(
|
||||
"INSERT INTO accounts (name, cui, active, status) VALUES (?, ?, ?, ?)",
|
||||
(name, cui, 1 if active else 0, "active" if active else "pending"),
|
||||
"INSERT INTO accounts (name, cui, email, active, status, tier, trial_until, "
|
||||
"requested_plan, consent_at) "
|
||||
"VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
||||
(name, cui, email, 1 if active else 0, "active" if active else "pending",
|
||||
"free", trial_until, req_plan, consent_at),
|
||||
)
|
||||
except sqlite3.IntegrityError:
|
||||
existing = conn.execute("SELECT id FROM accounts WHERE cui=?", (cui,)).fetchone()
|
||||
@@ -54,6 +89,21 @@ def create_account(
|
||||
return int(cur.lastrowid or 0)
|
||||
|
||||
|
||||
def account_is_complete(row: sqlite3.Row | dict) -> bool:
|
||||
"""Returneaza True daca contul are companie (name), email si CUI ne-goale.
|
||||
|
||||
Contul de sistem id=1 (default) este EXCEPTAT si returneaza intotdeauna True
|
||||
(nu are sens sa-l marcam ca incomplet — nu e un cont de client).
|
||||
"""
|
||||
acct_id = row["id"] if "id" in row.keys() else None
|
||||
if acct_id == 1:
|
||||
return True
|
||||
name = (row["name"] or "").strip()
|
||||
cui = (row["cui"] or "").strip()
|
||||
email_val = (row["email"] or "").strip() if "email" in row.keys() else ""
|
||||
return bool(name and cui and email_val)
|
||||
|
||||
|
||||
def set_active(conn: sqlite3.Connection, account_id: int, active: bool) -> None:
|
||||
"""Comuta `accounts.active`. Idempotent (set activ pe activ nu arunca).
|
||||
Cont inexistent -> ValueError.
|
||||
@@ -74,6 +124,8 @@ def set_active(conn: sqlite3.Connection, account_id: int, active: bool) -> None:
|
||||
# Stari de ciclu de viata gestionate explicit (5.5). 'deleted' = stergere soft (purjata de
|
||||
# retentie); restul sunt reversibile.
|
||||
VALID_STATUSES = ("pending", "active", "blocked", "archived", "deleted")
|
||||
# Tieruri de cont valide (5.17). Sursa de adevar: app/plans.py#PLANS (nu duplica valorile).
|
||||
VALID_TIERS = ("free", "standard", "pro", "premium")
|
||||
# Verbele care nu se pot aplica contului de sistem id=1 (protejat, ca la deactivate in 3.3b).
|
||||
_PROTECTED_ACCOUNT_ID = 1
|
||||
|
||||
@@ -98,6 +150,83 @@ def set_status(conn: sqlite3.Connection, account_id: int, status: str) -> None:
|
||||
)
|
||||
|
||||
|
||||
def set_tier(
|
||||
conn: sqlite3.Connection,
|
||||
account_id: int,
|
||||
tier: str,
|
||||
trial_until: str | None = None,
|
||||
) -> None:
|
||||
"""Seteaza planul unui cont (tier + trial_until).
|
||||
|
||||
tier invalid -> ValueError cu mesaj clar.
|
||||
Contul de sistem id=1 e protejat (ca set_status).
|
||||
Cont inexistent -> ValueError.
|
||||
Logheaza schimbarea in app_events (reuse observ.log_event, fara PII nou).
|
||||
|
||||
trial_until: string ISO UTC ("YYYY-MM-DD HH:MM:SS") sau None (sterge trial-ul).
|
||||
"""
|
||||
if tier not in VALID_TIERS:
|
||||
raise ValueError(
|
||||
f"tier invalid: {tier!r} (valid: {', '.join(VALID_TIERS)})"
|
||||
)
|
||||
row = conn.execute("SELECT 1 FROM accounts WHERE id=?", (account_id,)).fetchone()
|
||||
if not row:
|
||||
raise ValueError(f"cont inexistent: {account_id}")
|
||||
if account_id == _PROTECTED_ACCOUNT_ID:
|
||||
raise ValueError(
|
||||
"Contul default (id=1) nu poate fi mutat pe alt plan via CLI "
|
||||
"(cont de sistem, tratat coerent)."
|
||||
)
|
||||
conn.execute(
|
||||
"UPDATE accounts SET tier=?, trial_until=? WHERE id=?",
|
||||
(tier, trial_until, account_id),
|
||||
)
|
||||
# Audit in app_events (decizie PRD 5.17 US-008, fara PII nou)
|
||||
try:
|
||||
from .observ import log_event
|
||||
log_event(
|
||||
"plan_schimbare_tier",
|
||||
account_id=account_id,
|
||||
mesaj=f"tier -> {tier}",
|
||||
context={"tier": tier, "trial_until": trial_until},
|
||||
conn=conn,
|
||||
)
|
||||
except Exception: # noqa: BLE001 — jurnal best-effort (ca observ.log_event)
|
||||
pass
|
||||
|
||||
|
||||
def set_trial(conn: sqlite3.Connection, account_id: int, trial_until: str | None) -> None:
|
||||
"""Seteaza DOAR `trial_until` (acorda/prelungeste/sterge trial Pro), fara a atinge `tier`.
|
||||
|
||||
Trial Pro activ (trial_until in viitor) ridica planul efectiv la 'pro' (vezi
|
||||
plans.effective_tier), indiferent de tier-ul de baza. Folosit din panoul admin ca sa
|
||||
acorzi un trial fara a schimba tier-ul de baza (post-trial).
|
||||
|
||||
Contul de sistem id=1 e protejat. Cont inexistent -> ValueError.
|
||||
trial_until: string ISO UTC ("YYYY-MM-DD HH:MM:SS") sau None (sterge trial-ul).
|
||||
"""
|
||||
row = conn.execute("SELECT 1 FROM accounts WHERE id=?", (account_id,)).fetchone()
|
||||
if not row:
|
||||
raise ValueError(f"cont inexistent: {account_id}")
|
||||
if account_id == _PROTECTED_ACCOUNT_ID:
|
||||
raise ValueError("Contul default (id=1) nu poate primi trial (cont de sistem).")
|
||||
conn.execute(
|
||||
"UPDATE accounts SET trial_until=? WHERE id=?", (trial_until, account_id)
|
||||
)
|
||||
# Audit in app_events (best-effort, fara PII nou — ca set_tier).
|
||||
try:
|
||||
from .observ import log_event
|
||||
log_event(
|
||||
"plan_trial_setat",
|
||||
account_id=account_id,
|
||||
mesaj=f"trial_until -> {trial_until or 'NULL'}",
|
||||
context={"trial_until": trial_until},
|
||||
conn=conn,
|
||||
)
|
||||
except Exception: # noqa: BLE001 — jurnal best-effort (ca observ.log_event)
|
||||
pass
|
||||
|
||||
|
||||
def delete_account(conn: sqlite3.Connection, account_id: int) -> None:
|
||||
"""Stergere SOFT: randul ramane ca tombstone (status='deleted', scos din liste), DAR datele
|
||||
sensibile se purjeaza IMEDIAT (GDPR/L.142): credentialele RAR criptate sterse, cheile API
|
||||
@@ -121,7 +250,8 @@ def list_accounts(conn: sqlite3.Connection) -> list[dict]:
|
||||
"""Metadate conturi (FARA `rar_creds_enc`), ordonate dupa id. Exclude conturile 'deleted'
|
||||
(stergere soft -> invizibile in panou)."""
|
||||
rows = conn.execute(
|
||||
"SELECT id, name, cui, active, status, created_at FROM accounts "
|
||||
"SELECT id, name, cui, email, active, status, tier, trial_until, "
|
||||
"requested_plan, consent_at, created_at FROM accounts "
|
||||
"WHERE status != 'deleted' ORDER BY id"
|
||||
).fetchall()
|
||||
return [dict(r) for r in rows]
|
||||
|
||||
@@ -8,14 +8,13 @@ Endpointuri:
|
||||
POST /v1/import/{id}/commit — gate HARD + enqueue randuri ok + log atestare
|
||||
GET /v1/import/{id}/export-failed — CSV cu randuri esuate (needs_data/needs_mapping/needs_review)
|
||||
|
||||
Reguli cheie (plan §3.1-3.4, §12):
|
||||
- Issue 6: scrieri bulk in tranzactie explicita BEGIN IMMEDIATE...COMMIT + executemany.
|
||||
- Eng#5: already_sent lookup BATCH (IN chunk ~900), nu N+1.
|
||||
- OV-3: duplicate_in_file EXCLUSIV la preview/commit. NU atinge reconcile.py/worker.
|
||||
- Issue 1 (TOCTOU): commit per-rand cu ON CONFLICT(idempotency_key) DO NOTHING.
|
||||
- Issue 5a: import_rows.raw_json CRIPTAT Fernet.
|
||||
- Issue 5b: fuzzy coloane refoloseste mapping.normalize_for_match (DRY).
|
||||
- T4/D3: drift semnatura coloane -> NU aplica orb, cere re-confirmare.
|
||||
Reguli cheie:
|
||||
- Scrieri bulk in tranzactie explicita BEGIN IMMEDIATE...COMMIT + executemany.
|
||||
- already_sent lookup BATCH (IN chunk ~900), nu N+1.
|
||||
- duplicate_in_file EXCLUSIV la preview/commit. NU atinge reconcile.py/worker.
|
||||
- TOCTOU: commit per-rand cu ON CONFLICT(idempotency_key) DO NOTHING.
|
||||
- import_rows.raw_json CRIPTAT Fernet.
|
||||
- Drift semnatura coloane -> NU aplica orb, cere re-confirmare.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -30,8 +29,10 @@ from fastapi import APIRouter, Depends, HTTPException, UploadFile
|
||||
from fastapi.responses import StreamingResponse
|
||||
from pydantic import BaseModel, Field
|
||||
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from ... import errors
|
||||
from ...auth import resolve_account_id
|
||||
from ...auth import require_api_access, resolve_account_id
|
||||
from ...crypto import decrypt_creds, encrypt_creds
|
||||
from ...db import get_connection
|
||||
from ...idempotency import build_key, canonicalize_row
|
||||
@@ -60,7 +61,7 @@ router = APIRouter(prefix="/v1/import", tags=["import"])
|
||||
# Marimea maxima a unui chunk pentru IN(...) SQLite (limite SQLite ~999)
|
||||
_IN_CHUNK = 900
|
||||
|
||||
# Campuri canonice si sinonimele lor pentru sugestie fuzzy coloane (Issue 5b/Eng#4)
|
||||
# Campuri canonice si sinonimele lor pentru sugestie fuzzy coloane
|
||||
_CANONICAL_SYNONYMS: dict[str, list[str]] = {
|
||||
"vin": ["VIN", "Serie sasiu", "Sasiu", "Serie", "Numar sasiu", "Nr sasiu", "Chassis"],
|
||||
"nr_inmatriculare": ["Nr inmatriculare", "Numar inmatriculare", "Numar auto", "Nr auto", "Numar", "Nr"],
|
||||
@@ -93,7 +94,7 @@ def _fuzzy_suggest_column(
|
||||
) -> list[dict]:
|
||||
"""Sugereaza campuri canonice pentru o coloana din fisier.
|
||||
|
||||
Refoloseste normalize_for_match + rapidfuzz.fuzz.token_sort_ratio (Issue 5b/Eng#4).
|
||||
Refoloseste normalize_for_match + rapidfuzz.fuzz.token_sort_ratio.
|
||||
Intoarce [{camp_canonic, score}] sortat descrescator.
|
||||
"""
|
||||
from rapidfuzz import fuzz, process
|
||||
@@ -131,6 +132,7 @@ def _resolve_row_for_preview(
|
||||
override: dict[str, Any] | None = None,
|
||||
valid_codes: set[str] | None = None,
|
||||
text_rules: list[dict] | None = None,
|
||||
reviewed: bool = False,
|
||||
) -> dict[str, Any]:
|
||||
"""Rezolva un rand din import pentru preview: aplica mapare coloane + validare.
|
||||
|
||||
@@ -140,10 +142,10 @@ def _resolve_row_for_preview(
|
||||
errors: lista erori validare
|
||||
flags: motive needs_review
|
||||
|
||||
`override` (3.6, Approach B): patch CANONIC editat in preview, aplicat ULTIMUL
|
||||
peste valorile mapate (dupa `json_mapare` si canonicalizare). Permite corectarea
|
||||
unei valori sau completarea unui camp a carui coloana LIPSESTE din fisier, fara
|
||||
sa atinga `raw_json`/idempotency.
|
||||
`override`: patch CANONIC editat in preview, aplicat ULTIMUL peste valorile
|
||||
mapate (dupa `json_mapare` si canonicalizare). Permite corectarea unei valori
|
||||
sau completarea unui camp a carui coloana LIPSESTE din fisier, fara sa atinga
|
||||
`raw_json`/idempotency.
|
||||
"""
|
||||
# Aplica maparea de coloane
|
||||
mapped: dict[str, Any] = {}
|
||||
@@ -151,7 +153,7 @@ def _resolve_row_for_preview(
|
||||
if col_fisier in raw_row and camp_canonic:
|
||||
mapped[camp_canonic] = raw_row[col_fisier]
|
||||
|
||||
# Detectie coloane cu formule (Issue 3) — nu blocheaza, dar adauga flag
|
||||
# Detectie coloane cu formule — nu blocheaza, dar adauga flag
|
||||
formula_flag: list[str] = []
|
||||
for col_fisier, camp_canonic in json_mapare.items():
|
||||
if col_fisier in formula_columns:
|
||||
@@ -186,7 +188,15 @@ def _resolve_row_for_preview(
|
||||
denumire = str(denumire_val).strip() if denumire_val not in (None, "") else str(operatie_val)
|
||||
mapped["prestatii"] = [{"cod_op_service": str(operatie_val), "denumire": denumire}]
|
||||
|
||||
# Canonicalizare (T9): normalizeaza VIN/nr/odometru
|
||||
# obs derive-on-empty (D7/E3 PRD 5.15): daca obs e gol si avem operatie,
|
||||
# copiem denumirea operatiei in obs (nu o mutam — op_service ramane neatins).
|
||||
# DERIVE-ON-EMPTY: doar cand obs e gol, ca sa fie idempotent la re-preview/re-editare.
|
||||
obs_curent = str(mapped.get("obs") or "").strip()
|
||||
if not obs_curent and operatie_val:
|
||||
obs_denumire = str(denumire_val).strip() if denumire_val not in (None, "") else str(operatie_val)
|
||||
mapped["obs"] = obs_denumire
|
||||
|
||||
# Canonicalizare: normalizeaza VIN/nr/odometru
|
||||
canon = canonicalize_row(mapped)
|
||||
mapped.update({
|
||||
"vin": canon["vin"],
|
||||
@@ -194,7 +204,7 @@ def _resolve_row_for_preview(
|
||||
"odometru_final": canon["odometru_final"],
|
||||
})
|
||||
|
||||
# Override editat in preview (3.6) — aplicat ULTIMUL, peste valorile mapate +
|
||||
# Override editat in preview — aplicat ULTIMUL, peste valorile mapate +
|
||||
# canonicalizate. Valorile din override sunt deja canonice (vezi _merge_override).
|
||||
if override:
|
||||
mapped.update(override)
|
||||
@@ -221,8 +231,10 @@ def _resolve_row_for_preview(
|
||||
# Validare continut
|
||||
errors = validate_prezentare(mapped)
|
||||
|
||||
if all_flags:
|
||||
# needs_review: chiar daca validarea trece, flagurile blocheaza auto-send
|
||||
if all_flags and not reviewed:
|
||||
# needs_review: validarea a trecut, dar flagurile (date ambigue, formule) cer confirmare manuala.
|
||||
# Daca reviewed=True (operatorul a confirmat explicit valorile in modal), sarim
|
||||
# acest return si continuam spre ok/needs_data (US-007, PRD 5.12).
|
||||
return {
|
||||
"resolved_status": "needs_review",
|
||||
"resolved": mapped,
|
||||
@@ -230,14 +242,7 @@ def _resolve_row_for_preview(
|
||||
"flags": all_flags,
|
||||
}
|
||||
|
||||
# auto_send gate (T6/OV-1)
|
||||
if has_no_auto_send(resolved, mapping_meta):
|
||||
return {
|
||||
"resolved_status": "needs_mapping",
|
||||
"resolved": mapped,
|
||||
"errors": [{"auto_send": "cod mapat cu auto_send=0; review manual inainte de trimitere"}],
|
||||
"flags": all_flags,
|
||||
}
|
||||
# US-001 (PRD 5.11): ramura auto_send eliminata din preview.
|
||||
|
||||
if errors:
|
||||
return {
|
||||
@@ -261,9 +266,10 @@ def _build_idempotency_key(account_id: int | None, resolved: dict[str, Any]) ->
|
||||
return build_key(account_id, canon)
|
||||
|
||||
|
||||
# Campuri de continut editabile in preview (3.6). Operatia/codul RAR NU se editeaza
|
||||
# aici (raman in panoul de mapare) — vezi Non-Goals din PRD 3.6.
|
||||
EDIT_FIELDS = ("vin", "nr_inmatriculare", "data_prestatie", "odometru_initial", "odometru_final")
|
||||
# Campuri de continut editabile in preview. Operatia/codul RAR NU se editeaza
|
||||
# aici (raman in panoul de mapare). obs = text liber, se trateaza ca non-canonic
|
||||
# (doar .strip(), fara canonicalize_row) — urmeaza ramura `else` din _merge_override.
|
||||
EDIT_FIELDS = ("vin", "nr_inmatriculare", "data_prestatie", "odometru_initial", "odometru_final", "obs")
|
||||
|
||||
|
||||
def _merge_override(current: dict[str, Any], fields: dict[str, str | None]) -> dict[str, Any]:
|
||||
@@ -284,7 +290,15 @@ def _merge_override(current: dict[str, Any], fields: dict[str, str | None]) -> d
|
||||
continue
|
||||
s = str(val).strip()
|
||||
if s == "":
|
||||
out.pop(camp, None) # empty = clear
|
||||
if camp == "obs":
|
||||
# obs e camp DERIVAT (copiaza denumirea operatiei cand e gol). Empty =
|
||||
# STERGERE EXPLICITA a userului -> pastram obs='' in override ca
|
||||
# derive-on-empty sa NU il re-deriveze (override aplicat ULTIMUL
|
||||
# suprascrie derivarea, in preview si la commit). Un pop ar fi pierdut
|
||||
# semnalul "sters explicit" -> obs re-derivat silentios din denumire.
|
||||
out["obs"] = ""
|
||||
else:
|
||||
out.pop(camp, None) # empty = clear (revine la valoarea din fisier)
|
||||
else:
|
||||
raw[camp] = s
|
||||
if raw:
|
||||
@@ -345,12 +359,15 @@ def apply_row_override(
|
||||
|
||||
new_override = _merge_override(current, fields)
|
||||
enc = encrypt_creds(new_override) if new_override else None
|
||||
conn.execute("UPDATE import_rows SET override_json=? WHERE id=?", (enc, row["rid"]))
|
||||
# D#9 (PRD 5.12): resetam reviewed=0 la orice schimbare de valoare — operatorul
|
||||
# trebuie sa reconfirme dupa editare. NU conditionam pe reviewed curent: orice override
|
||||
# (chiar si revert la valoarea initiala) anuleaza confirmarea implicita.
|
||||
conn.execute("UPDATE import_rows SET override_json=?, reviewed=0 WHERE id=?", (enc, row["rid"]))
|
||||
return new_override
|
||||
|
||||
|
||||
def _already_sent_lookup(conn, account_id: int, keys: list[str]) -> dict[str, dict]:
|
||||
"""Cauta cheile de idempotenta in submissions (batch, nu N+1 — Eng#5).
|
||||
"""Cauta cheile de idempotenta in submissions (batch, nu N+1).
|
||||
|
||||
Intoarce {idempotency_key: {id, id_prezentare, created_at}} pentru cheile gasite.
|
||||
"""
|
||||
@@ -371,7 +388,7 @@ def _already_sent_lookup(conn, account_id: int, keys: list[str]) -> dict[str, di
|
||||
"id_prezentare": r["id_prezentare"],
|
||||
"created_at": r["created_at"],
|
||||
}
|
||||
# Dual-lookup pentru chei legacy (OV-2: chei vechi cu account_id=None)
|
||||
# Dual-lookup pentru chei legacy (chei vechi cu account_id=None)
|
||||
legacy_keys_needed = [k for k in chunk if k not in found]
|
||||
if legacy_keys_needed:
|
||||
lph = ",".join("?" * len(legacy_keys_needed))
|
||||
@@ -398,13 +415,12 @@ def _already_sent_lookup(conn, account_id: int, keys: list[str]) -> dict[str, di
|
||||
async def upload_import(
|
||||
file: UploadFile,
|
||||
sheet_name: str | None = None,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
account_id: int = Depends(require_api_access),
|
||||
) -> dict:
|
||||
"""Upload fisier xlsx/csv -> staging in import_batches/import_rows.
|
||||
|
||||
Nu trimite nimic la RAR. Intoarce {import_id, columns, sample_rows, sheets?}.
|
||||
PII (raw_json) criptat Fernet la rest (Issue 5a).
|
||||
Scrieri bulk in tranzactie explicita (Issue 6).
|
||||
PII (raw_json) criptat Fernet la rest. Scrieri bulk in tranzactie explicita.
|
||||
"""
|
||||
acct = account_or_default(account_id)
|
||||
data = await file.read()
|
||||
@@ -468,7 +484,7 @@ async def upload_import(
|
||||
try:
|
||||
sig = _signature(parsed.columns)
|
||||
|
||||
# Issue 6: tranzactie explicita BEGIN IMMEDIATE + executemany
|
||||
# Tranzactie explicita BEGIN IMMEDIATE + executemany
|
||||
conn.execute("BEGIN IMMEDIATE")
|
||||
try:
|
||||
# Insert import_batches
|
||||
@@ -482,7 +498,7 @@ async def upload_import(
|
||||
# Insert import_rows bulk (executemany) cu PII criptat
|
||||
row_params = []
|
||||
for i, row_dict in enumerate(parsed.rows):
|
||||
raw_json_enc = encrypt_creds(row_dict) # Criptat Fernet (Issue 5a)
|
||||
raw_json_enc = encrypt_creds(row_dict) # Criptat Fernet
|
||||
row_params.append((batch_id, i, raw_json_enc, "pending", None))
|
||||
|
||||
conn.executemany(
|
||||
@@ -506,11 +522,8 @@ async def upload_import(
|
||||
# Sample rows (primele 3, fara PII)
|
||||
sample = parsed.rows[:3]
|
||||
|
||||
# Persistam metadata parsedata (coercion_flags, date_col_format, formula_columns)
|
||||
# in import_batches pentru refolosire la preview (stocam ca JSON in 'status' nu e OK,
|
||||
# ci ca metadate suplimentare — le stocam intr-un rand separat sau returnam direct)
|
||||
# Solutie: le returnam in raspuns; preview-ul le va recalcula din raw_json deja stocat
|
||||
# SAU le stocam ca un camp extra. Cel mai simplu: stocam coloanele in batch.
|
||||
# Metadata parsata (coercion_flags etc.) se intoarce in raspuns; preview-ul
|
||||
# o recalculeaza din raw_json deja stocat.
|
||||
conn.execute(
|
||||
"UPDATE import_batches SET ok=?, needs_review=? WHERE id=?",
|
||||
(0, len(parsed.coercion_flags), batch_id),
|
||||
@@ -532,7 +545,7 @@ async def upload_import(
|
||||
result["column_mapping"] = json.loads(existing_mapping["json_mapare"])
|
||||
result["format_data"] = existing_mapping["format_data"]
|
||||
else:
|
||||
# Sugestii fuzzy per coloana (Issue 5b: refoloseste normalize_for_match)
|
||||
# Sugestii fuzzy per coloana
|
||||
suggestions: dict[str, list[dict]] = {}
|
||||
for col in parsed.columns:
|
||||
sugg = _fuzzy_suggest_column(col, limit=3)
|
||||
@@ -676,7 +689,7 @@ def save_column_mapping(
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# GET /v1/import/{id}/preview — 6 stari per rand (T2 + T11) #
|
||||
# GET /v1/import/{id}/preview — 6 stari per rand #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
@router.get("/{import_id}/preview")
|
||||
@@ -686,8 +699,8 @@ def preview_import(
|
||||
) -> dict:
|
||||
"""Preview 6 stari per rand: ok/needs_mapping/needs_data/needs_review/already_sent/duplicate_in_file.
|
||||
|
||||
Nu enqueue-aza nimic. Already_sent = lookup batch (Eng#5). Duplicate_in_file = intra-batch
|
||||
collision (OV-3: EXCLUSIV aici, NU in reconcile.py/worker).
|
||||
Nu enqueue-aza nimic. Already_sent = lookup batch. Duplicate_in_file = intra-batch
|
||||
collision (EXCLUSIV aici, NU in reconcile.py/worker).
|
||||
"""
|
||||
acct = account_or_default(account_id)
|
||||
conn = get_connection()
|
||||
@@ -708,7 +721,7 @@ def preview_import(
|
||||
if not raw_rows_db:
|
||||
return {"rows": [], "summary": {}}
|
||||
|
||||
# Decripteaza si reconstruieste randurile + override-urile editate (3.6)
|
||||
# Decripteaza si reconstruieste randurile + override-urile editate
|
||||
rows: list[dict] = []
|
||||
overrides: list[dict] = []
|
||||
for r in raw_rows_db:
|
||||
@@ -747,22 +760,18 @@ def preview_import(
|
||||
json_mapare: dict[str, str] = json.loads(mapping_row["json_mapare"])
|
||||
format_data = mapping_row["format_data"]
|
||||
|
||||
# Incarca maparea de operatii o singura data (Eng#5: load_mapping o singura data)
|
||||
# Incarca maparea de operatii o singura data
|
||||
mapping_meta = load_mapping_meta(conn, acct)
|
||||
mapping = {op: meta["cod_prestatie"] for op, meta in mapping_meta.items()}
|
||||
# T2: validare nomenclator + reguli text incarcate O DATA, inainte de bucla pe randuri.
|
||||
# Validare nomenclator + reguli text incarcate O DATA, inainte de bucla pe randuri.
|
||||
valid_codes = load_nomenclator_codes(conn) or None
|
||||
text_rules = load_text_rules(conn, acct)
|
||||
|
||||
# Reconstruieste parsed info (coercion_flags si date_col_format) din datele stocate
|
||||
# Nota: import_rows stocheaza raw_json dupa coercion (din parse_file)
|
||||
# Recalculam flags din valorile stocate (coercion_flags nu e stocat separat)
|
||||
# Vom folosi o detectie simpla: VIN-uri care par numerice si odometru float
|
||||
# Recalculam coercion_flags din valorile stocate (nu sunt persistate separat):
|
||||
# detectie simpla de VIN numeric.
|
||||
coercion_flags_map: dict[int, list[str]] = {}
|
||||
# Detectam din valorile stocate
|
||||
for i, row_dict in enumerate(rows):
|
||||
flags = []
|
||||
# Detectam VIN numeric: daca valoarea a fost stocata si arata ca numar
|
||||
for col_f, camp_c in json_mapare.items():
|
||||
if camp_c == "vin":
|
||||
vin_val = row_dict.get(col_f)
|
||||
@@ -830,11 +839,11 @@ def preview_import(
|
||||
"idempotency_key": key,
|
||||
})
|
||||
|
||||
# Already_sent: batch lookup (Eng#5 — nu N+1)
|
||||
# Already_sent: batch lookup (nu N+1)
|
||||
unique_keys = list(set(keys_for_lookup))
|
||||
already_sent_map = _already_sent_lookup(conn, account_id, unique_keys)
|
||||
|
||||
# Duplicate_in_file (OV-3): detectie coliziuni intra-batch
|
||||
# Duplicate_in_file: detectie coliziuni intra-batch.
|
||||
# Grupam pe cheie de idempotenta: >1 rand cu aceeasi cheie = duplicate
|
||||
key_to_indices: dict[str, list[int]] = {}
|
||||
for row in preview_rows:
|
||||
@@ -857,7 +866,7 @@ def preview_import(
|
||||
row["already_sent_info"] = sent_info
|
||||
continue
|
||||
|
||||
# Duplicate_in_file (OV-3): >1 rand cu aceeasi cheie in ACELASI fisier
|
||||
# Duplicate_in_file: >1 rand cu aceeasi cheie in ACELASI fisier
|
||||
indices_with_same_key = key_to_indices.get(k, [])
|
||||
if len(indices_with_same_key) > 1 and row["resolved_status"] in ("ok", "needs_review", "needs_data"):
|
||||
others = [idx for idx in indices_with_same_key if idx != row["row_index"]]
|
||||
@@ -911,7 +920,7 @@ def preview_import(
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# POST /v1/import/{id}/commit — gate HARD + enqueue + log atestare (T5+T12) #
|
||||
# POST /v1/import/{id}/commit — gate HARD + enqueue + log atestare #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
class CommitIn(BaseModel):
|
||||
@@ -927,11 +936,11 @@ class CommitIn(BaseModel):
|
||||
def commit_import(
|
||||
import_id: int,
|
||||
req: CommitIn,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
account_id: int = Depends(require_api_access),
|
||||
) -> dict:
|
||||
"""Gate HARD confirmare + enqueue randuri ok + log atestare (T5+T12).
|
||||
"""Gate HARD confirmare + enqueue randuri ok + log atestare.
|
||||
|
||||
TOCTOU (Issue 1): INSERT per-rand cu ON CONFLICT(idempotency_key) DO NOTHING.
|
||||
TOCTOU: INSERT per-rand cu ON CONFLICT(idempotency_key) DO NOTHING.
|
||||
Randuri colidante = reclasificate already_sent in rezultatul commit-ului.
|
||||
rows_hash + n_confirmed acopera DOAR randurile efectiv puse in coada.
|
||||
"""
|
||||
@@ -948,10 +957,30 @@ def commit_import(
|
||||
if batch["status"] == "committed":
|
||||
raise HTTPException(status_code=409, detail="batch deja comis")
|
||||
|
||||
# Incarca randurile cu stare ok sau needs_review
|
||||
# D#8 (PRD 5.12): gate commit derivat din DB `reviewed` pe AMBELE canale.
|
||||
# API: reviewed_rows pastrat (contract stabil) dar seteaza reviewed=1 in DB inainte
|
||||
# de interogare. Randurile needs_review cu reviewed=1 sunt incluse in comit.
|
||||
if req.reviewed_rows:
|
||||
conn.execute("BEGIN IMMEDIATE")
|
||||
try:
|
||||
for idx in req.reviewed_rows:
|
||||
conn.execute(
|
||||
"UPDATE import_rows SET reviewed=1 "
|
||||
"WHERE batch_id=? AND row_index=? AND resolved_status='needs_review'",
|
||||
(import_id, idx),
|
||||
)
|
||||
conn.execute("COMMIT")
|
||||
except Exception:
|
||||
conn.execute("ROLLBACK")
|
||||
raise
|
||||
|
||||
# Incarca randurile ok + needs_review confirmate (reviewed=1)
|
||||
ok_rows_db = conn.execute(
|
||||
"SELECT row_index, raw_json, override_json, resolved_status FROM import_rows "
|
||||
"WHERE batch_id=? AND resolved_status IN ('ok', 'needs_review') ORDER BY row_index",
|
||||
"SELECT row_index, raw_json, override_json, resolved_status, reviewed "
|
||||
"FROM import_rows "
|
||||
"WHERE batch_id=? AND (resolved_status='ok' OR "
|
||||
"(resolved_status='needs_review' AND reviewed=1)) "
|
||||
"ORDER BY row_index",
|
||||
(import_id,),
|
||||
).fetchall()
|
||||
|
||||
@@ -963,8 +992,6 @@ def commit_import(
|
||||
|
||||
# Decripteaza randurile ok
|
||||
ok_rows: list[dict] = []
|
||||
ok_indices: list[int] = []
|
||||
review_indices: set[int] = set()
|
||||
|
||||
for r in ok_rows_db:
|
||||
try:
|
||||
@@ -973,28 +1000,12 @@ def commit_import(
|
||||
continue
|
||||
except Exception:
|
||||
continue
|
||||
|
||||
if r["resolved_status"] == "ok":
|
||||
ok_rows.append({"row_index": r["row_index"], "data": row_data,
|
||||
"override": _override_of(r), "status": "ok"})
|
||||
ok_indices.append(r["row_index"])
|
||||
elif r["resolved_status"] == "needs_review":
|
||||
review_indices.add(r["row_index"])
|
||||
|
||||
# needs_review bifate explicit (Voce#1 — atestare pe valori)
|
||||
confirmed_review = [idx for idx in req.reviewed_rows if idx in review_indices]
|
||||
for idx in confirmed_review:
|
||||
# Gaseste randul needs_review si il adauga la ok_rows
|
||||
for r in ok_rows_db:
|
||||
if r["row_index"] == idx and r["resolved_status"] == "needs_review":
|
||||
try:
|
||||
row_data = decrypt_creds(r["raw_json"])
|
||||
if row_data:
|
||||
ok_rows.append({"row_index": idx, "data": row_data,
|
||||
"override": _override_of(r), "status": "needs_review"})
|
||||
ok_indices.append(idx)
|
||||
except Exception:
|
||||
pass
|
||||
ok_rows.append({
|
||||
"row_index": r["row_index"],
|
||||
"data": row_data,
|
||||
"override": _override_of(r),
|
||||
"status": r["resolved_status"],
|
||||
})
|
||||
|
||||
# Gate HARD: n_confirmat trebuie sa fie EXACT egal cu numarul de randuri ok
|
||||
n_total_ok = len(ok_rows)
|
||||
@@ -1013,6 +1024,48 @@ def commit_import(
|
||||
if n_total_ok == 0:
|
||||
raise HTTPException(status_code=422, detail="Niciun rand ok de confirmat.")
|
||||
|
||||
# T3 (PRD 5.17): enforce volum plan — INAINTE de enqueue (invariant idempotenta).
|
||||
# Decizie #21: respingere TOTALA a lotului (nu enqueue partial tacut).
|
||||
from ...config import get_settings as _get_settings
|
||||
from ...plans import PLANS, effective_tier, monthly_usage
|
||||
from ...observ import log_event as _log_event_plan
|
||||
_settings = _get_settings()
|
||||
if _settings.enforce_plans:
|
||||
_acct_row = conn.execute(
|
||||
"SELECT tier, trial_until FROM accounts WHERE id=?", (acct,)
|
||||
).fetchone()
|
||||
_now = datetime.now(timezone.utc)
|
||||
_et = effective_tier(_acct_row, _now)
|
||||
_plan_limit = PLANS[_et].get("monthly_limit")
|
||||
if _plan_limit is not None:
|
||||
_usage = monthly_usage(conn, acct, _now)
|
||||
if _usage + n_total_ok > _plan_limit:
|
||||
_remaining = max(0, _plan_limit - _usage)
|
||||
_log_event_plan(
|
||||
"plan_limita_lunara_atinsa",
|
||||
account_id=acct,
|
||||
nivel="WARNING",
|
||||
mesaj=f"Import de {n_total_ok} respins (usage={_usage}, limita={_plan_limit})",
|
||||
context={
|
||||
"n_to_enqueue": n_total_ok, "usage": _usage,
|
||||
"plan_limit": _plan_limit, "tier": _et,
|
||||
},
|
||||
conn=conn,
|
||||
)
|
||||
raise HTTPException(
|
||||
status_code=422,
|
||||
detail={
|
||||
"error": "plan_limita_lunara",
|
||||
**errors.eroare(
|
||||
"PLAN_LIMITA_LUNARA",
|
||||
cauza=(
|
||||
f"Ai trimis {_usage}/{_plan_limit} prezentari luna aceasta;"
|
||||
f" mai poti trimite {_remaining}."
|
||||
),
|
||||
),
|
||||
},
|
||||
)
|
||||
|
||||
# Incarca maparea de coloane pentru a construi payload-ul
|
||||
first_row_db = conn.execute(
|
||||
"SELECT raw_json FROM import_rows WHERE batch_id=? ORDER BY row_index LIMIT 1",
|
||||
@@ -1040,7 +1093,7 @@ def commit_import(
|
||||
# Incarca maparea de operatii
|
||||
mapping_meta = load_mapping_meta(conn, acct)
|
||||
mapping = {op: meta["cod_prestatie"] for op, meta in mapping_meta.items()}
|
||||
# T2: validare nomenclator + reguli text incarcate O DATA, inainte de bucla pe randuri.
|
||||
# Validare nomenclator + reguli text incarcate O DATA, inainte de bucla pe randuri.
|
||||
valid_codes = load_nomenclator_codes(conn) or None
|
||||
text_rules = load_text_rules(conn, acct)
|
||||
|
||||
@@ -1049,10 +1102,9 @@ def commit_import(
|
||||
toctou_collisions: list[int] = []
|
||||
rows_for_hash: list[str] = []
|
||||
|
||||
# Enqueue in tranzactie explicita (Issue 6)
|
||||
# Enqueue in tranzactie explicita
|
||||
conn.execute("BEGIN IMMEDIATE")
|
||||
try:
|
||||
# purge_after pentru submissions noi (T16)
|
||||
purge_after_sql = "datetime('now', '+90 days')"
|
||||
|
||||
for ok_row in ok_rows:
|
||||
@@ -1087,6 +1139,13 @@ def commit_import(
|
||||
denumire = str(denumire_val).strip() if denumire_val not in (None, "") else str(operatie_val)
|
||||
mapped["prestatii"] = [{"cod_op_service": str(operatie_val), "denumire": denumire}]
|
||||
|
||||
# obs derive-on-empty (D7/E3 PRD 5.15): copiere denumire in obs daca obs e gol.
|
||||
# Identic cu logica din _resolve_row_for_preview (override aplicat tot ultimul).
|
||||
obs_curent = str(mapped.get("obs") or "").strip()
|
||||
if not obs_curent and operatie_val:
|
||||
obs_denumire = str(denumire_val).strip() if denumire_val not in (None, "") else str(operatie_val)
|
||||
mapped["obs"] = obs_denumire
|
||||
|
||||
# Rezolva prestatii INAINTE de canonicalizare (altfel cheia difera de cea din preview)
|
||||
prestatii = mapped.get("prestatii") or []
|
||||
resolved, _ = resolve_prestatii(prestatii, mapping, valid_codes, text_rules)
|
||||
@@ -1100,7 +1159,7 @@ def commit_import(
|
||||
"odometru_final": canon["odometru_final"],
|
||||
})
|
||||
|
||||
# Override editat in preview (3.6) — aplicat ULTIMUL, ca in resolver.
|
||||
# Override editat in preview — aplicat ULTIMUL, ca in resolver.
|
||||
override = ok_row.get("override") or {}
|
||||
if override:
|
||||
mapped.update(override)
|
||||
@@ -1127,7 +1186,7 @@ def commit_import(
|
||||
|
||||
payload_json = json.dumps(mapped, ensure_ascii=False)
|
||||
|
||||
# INSERT ON CONFLICT DO NOTHING (TOCTOU — Issue 1)
|
||||
# INSERT ON CONFLICT DO NOTHING (TOCTOU)
|
||||
cur = conn.execute(
|
||||
"INSERT OR IGNORE INTO submissions "
|
||||
"(idempotency_key, account_id, status, payload_json, batch_id, row_index, purge_after) "
|
||||
@@ -1140,7 +1199,6 @@ def commit_import(
|
||||
toctou_collisions.append(row_index)
|
||||
else:
|
||||
sub_id = cur.lastrowid
|
||||
# US-010: telemetrie pentru itemii rezolvati prin regula text.
|
||||
_emite_text_rule_hits(conn, acct, int(sub_id), resolved)
|
||||
enqueued.append({
|
||||
"submission_id": sub_id,
|
||||
@@ -1155,7 +1213,7 @@ def commit_import(
|
||||
|
||||
n_enqueued = len(enqueued)
|
||||
|
||||
# Log atestare (Voce#9): rows_hash + n_confirmed acopera DOAR randurile puse in coada
|
||||
# Log atestare: rows_hash + n_confirmed acopera DOAR randurile puse in coada
|
||||
rows_hash = hashlib.sha256(
|
||||
json.dumps(sorted(rows_for_hash), ensure_ascii=False).encode("utf-8")
|
||||
).hexdigest() if rows_for_hash else ""
|
||||
@@ -1185,17 +1243,19 @@ def commit_import(
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# POST /v1/import/{id}/rand/{row_index}/editeaza — editare celule preview (3.6) #
|
||||
# POST /v1/import/{id}/rand/{row_index}/editeaza — editare celule preview #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
class RandEditIn(BaseModel):
|
||||
"""Campuri de continut editabile in preview. None = ne-trimis (neschimbat);
|
||||
"" = sterge override-ul (revine la valoarea din fisier)."""
|
||||
"" = sterge override-ul (revine la valoarea din fisier).
|
||||
obs = text liber fara validare de continut (US-005 PRD 5.15)."""
|
||||
vin: str | None = None
|
||||
nr_inmatriculare: str | None = None
|
||||
data_prestatie: str | None = None
|
||||
odometru_initial: str | None = None
|
||||
odometru_final: str | None = None
|
||||
obs: str | None = None
|
||||
|
||||
|
||||
@router.post("/{import_id}/rand/{row_index}/editeaza")
|
||||
@@ -1205,7 +1265,7 @@ def editeaza_rand(
|
||||
req: RandEditIn,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
) -> dict:
|
||||
"""Persista editarea unui rand de preview (mutatie pura — Approach B, 3.6).
|
||||
"""Persista editarea unui rand de preview (mutatie pura).
|
||||
|
||||
NU recalculeaza statusul si NU atinge `submissions`; preview-ul rederiva statusul
|
||||
prin `_resolve_row_for_preview` cu override aplicat ultimul.
|
||||
@@ -1225,7 +1285,7 @@ def editeaza_rand(
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# GET /v1/import/{id}/export-failed — CSV randuri esuate (T8) #
|
||||
# GET /v1/import/{id}/export-failed — CSV randuri esuate #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
_EXPORT_FAILED_COLUMNS = [
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
"""Router integrare US-001 — endpoint-uri de integrare externe.
|
||||
"""Router integrare — endpoint-uri de integrare externe.
|
||||
|
||||
Endpointuri:
|
||||
GET /v1/ping — readiness check per cont (autentificat sau dev fallback)
|
||||
|
||||
@@ -1,12 +1,10 @@
|
||||
"""API v1 — suprafata gateway (schelet).
|
||||
"""API v1 — suprafata gateway.
|
||||
|
||||
Endpointuri din plan.md sect. 4. In schelet:
|
||||
Endpointuri:
|
||||
- POST /v1/prezentari: enqueue cu idempotenta (dedup pe idempotency_key UNIQUE).
|
||||
- GET /v1/prezentari, /v1/prezentari/{id}: monitorizare coada.
|
||||
- GET /v1/nomenclator: cache local.
|
||||
- GET /v1/mapari: listare mapari cont.
|
||||
Validarea completa (T3), maparea op->cod, auth API-key, redactarea creds in
|
||||
middleware (CORE) si exportul CSV vin ulterior — marcate TODO unde lipsesc.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -15,11 +13,13 @@ import csv
|
||||
import io
|
||||
import json
|
||||
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from fastapi import APIRouter, Depends, HTTPException
|
||||
from fastapi.responses import StreamingResponse
|
||||
from pydantic import BaseModel, Field
|
||||
|
||||
from ...auth import resolve_account_id
|
||||
from ...auth import require_api_access, resolve_account_id
|
||||
from ...crypto import encrypt_creds
|
||||
from ...db import get_connection
|
||||
from ...errors import eroare as err_eroare
|
||||
@@ -79,7 +79,7 @@ def _classify_modal(content, mapping, mapping_meta, valid_codes, error_mode, tex
|
||||
|
||||
|
||||
def _erori_nemapate(unmapped: list[dict]) -> list[dict]:
|
||||
"""Coduri nemapate imbogatite cu 3 niveluri (COD_NEMAPAT), pentru raspuns onest."""
|
||||
"""Coduri nemapate imbogatite cu 3 niveluri (COD_NEMAPAT)."""
|
||||
return [
|
||||
{**u, **err_eroare("COD_NEMAPAT", cauza=f"cod {u.get('cod_op_service')} necunoscut/fara mapare RAR")}
|
||||
for u in unmapped
|
||||
@@ -87,10 +87,12 @@ def _erori_nemapate(unmapped: list[dict]) -> list[dict]:
|
||||
|
||||
|
||||
def _motiv_clasificare(cl: dict) -> str | None:
|
||||
"""Rezumat uman pe o linie pentru un rezultat de clasificare (PRD 5.7).
|
||||
"""Rezumat uman pe o linie pentru un rezultat de clasificare.
|
||||
|
||||
None cand status='queued'. Acopera toate ramurile de blocaj: erori de continut
|
||||
(needs_data), coduri nemapate (needs_mapping) si auto_send oprit (needs_mapping).
|
||||
None cand status='queued'. Acopera ramurile de blocaj: erori de continut
|
||||
(needs_data) si coduri fara mapare RAR (needs_mapping).
|
||||
Dupa US-001: needs_mapping apare EXCLUSIV cand unmapped e non-gol
|
||||
(ramura auto_send_oprit era inaccesibila si a fost eliminata).
|
||||
"""
|
||||
if cl["status"] == "queued":
|
||||
return None
|
||||
@@ -101,13 +103,11 @@ def _motiv_clasificare(cl: dict) -> str | None:
|
||||
if cl["unmapped"]:
|
||||
coduri = ", ".join((u.get("cod_op_service") or "") for u in cl["unmapped"])
|
||||
return f"Coduri fara mapare RAR: {coduri}"
|
||||
if cl["status"] == "needs_mapping":
|
||||
return "Cod cu trimitere automata oprita; confirmare manuala inainte de trimitere."
|
||||
return None
|
||||
|
||||
|
||||
def _rezultat_enqueue(submission_id: int | None, cl: dict, **extra) -> SubmissionResult:
|
||||
"""SubmissionResult onest dintr-un rezultat de clasificare (PRD 5.7).
|
||||
"""SubmissionResult onest dintr-un rezultat de clasificare.
|
||||
|
||||
Populeaza erori (validare continut), nemapate (coduri fara mapare) si motiv (uman)
|
||||
pentru orice status != 'queued'. Aditiv: pe 'queued' toate raman goale/None.
|
||||
@@ -137,46 +137,84 @@ def _rezultat_respins(submission_id: int | None, cl: dict) -> SubmissionResult:
|
||||
@router.post("/prezentari", response_model=PrezentariResponse)
|
||||
def create_prezentari(
|
||||
req: PrezentareRequest,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
account_id: int = Depends(require_api_access),
|
||||
) -> PrezentariResponse:
|
||||
"""Enqueue una/mai multe prezentari. Idempotent: continut identic -> acelasi submission.
|
||||
|
||||
Validarea de continut (T3, app.validation) ruleaza inainte de enqueue:
|
||||
esecurile NU resping cererea, ci enqueue-aza cu status `needs_data` + motiv
|
||||
(plan.md sect. 3). JSON malformat -> 422 din Pydantic (validare de shape).
|
||||
Validarea de continut (app.validation) ruleaza inainte de enqueue: esecurile NU
|
||||
resping cererea, ci enqueue-aza cu status `needs_data` + motiv. JSON malformat ->
|
||||
422 din Pydantic (validare de shape).
|
||||
account_id vine din cheia API (resolve_account_id): cont real cu cheie,
|
||||
implicit id=1 in dev fara cheie, 401 fara cheie valida in prod.
|
||||
Nota: rar_credentials NU se persista (zero-storage) — worker-ul le va primi
|
||||
pe alt canal (T2); in schelet enqueue-ul doar stocheaza prezentarea.
|
||||
Cand rar_credentials lipseste, submission-ul intra fara creds efemere: worker-ul
|
||||
cade pe creds-urile durabile ale contului (`accounts.rar_creds_enc`).
|
||||
"""
|
||||
acct = account_or_default(account_id)
|
||||
# Creds RAR efemere: criptate si lipite de fiecare submission nou pana la
|
||||
# primul login reusit pentru cont (worker le sterge atunci). Zero-storage at
|
||||
# rest — niciodata in clar in DB/loguri (plan sect. 5). Optional: cand lipsesc,
|
||||
# rest — niciodata in clar in DB/loguri. Optional: cand lipsesc,
|
||||
# creds_enc=NULL si worker-ul foloseste creds-urile durabile ale contului.
|
||||
creds_enc = encrypt_creds(req.rar_credentials.model_dump()) if req.rar_credentials else None
|
||||
conn = get_connection()
|
||||
results: list[SubmissionResult] = []
|
||||
try:
|
||||
# T6/OV-1: load_mapping_meta include auto_send per op (gate pentru coduri noi).
|
||||
# load_mapping_meta incarca maparea op->cod RAR; dupa US-001, auto_send nu mai tine randuri.
|
||||
mapping_meta = load_mapping_meta(conn, acct)
|
||||
mapping = {op: meta["cod_prestatie"] for op, meta in mapping_meta.items()}
|
||||
# Validare cod_prestatie fata de nomenclator + modul la cod necunoscut/nemapat.
|
||||
# valid_codes gol (nomenclator nepopulat) -> None (nu validam, ca sa nu blocam tot).
|
||||
valid_codes = load_nomenclator_codes(conn) or None
|
||||
# Reguli text incarcate o data per cerere (seam partajat cu dry-run, invariant 5.2).
|
||||
# Reguli text incarcate o data per cerere (seam partajat cu dry-run).
|
||||
text_rules = load_text_rules(conn, acct)
|
||||
error_mode = _effective_on_unmapped_error(conn, acct, req.on_unmapped_error)
|
||||
|
||||
# T3 (PRD 5.17): enforce volum plan — INAINTE de build_key/enqueue (invariant idempotenta).
|
||||
# Decizie #21: respingere TOTALA a lotului (nu enqueue partial tacut).
|
||||
from ...config import get_settings as _get_settings
|
||||
from ...plans import PLANS, effective_tier, monthly_usage
|
||||
_settings = _get_settings()
|
||||
if _settings.enforce_plans:
|
||||
_acct_row = conn.execute(
|
||||
"SELECT tier, trial_until FROM accounts WHERE id=?", (acct,)
|
||||
).fetchone()
|
||||
_now = datetime.now(timezone.utc)
|
||||
_et = effective_tier(_acct_row, _now)
|
||||
_plan_limit = PLANS[_et].get("monthly_limit")
|
||||
if _plan_limit is not None:
|
||||
_usage = monthly_usage(conn, acct, _now)
|
||||
_nr_cerut = len(req.prezentari)
|
||||
if _usage + _nr_cerut > _plan_limit:
|
||||
_remaining = max(0, _plan_limit - _usage)
|
||||
log_event(
|
||||
"plan_limita_lunara_atinsa",
|
||||
account_id=acct,
|
||||
nivel="WARNING",
|
||||
mesaj=f"Lot de {_nr_cerut} respins (usage={_usage}, limita={_plan_limit})",
|
||||
context={
|
||||
"nr_cerut": _nr_cerut, "usage": _usage,
|
||||
"plan_limit": _plan_limit, "tier": _et,
|
||||
},
|
||||
conn=conn,
|
||||
)
|
||||
raise HTTPException(
|
||||
status_code=422,
|
||||
detail=err_eroare(
|
||||
"PLAN_LIMITA_LUNARA",
|
||||
cauza=(
|
||||
f"Ai trimis {_usage}/{_plan_limit} prezentari luna aceasta;"
|
||||
f" mai poti trimite {_remaining}."
|
||||
),
|
||||
),
|
||||
)
|
||||
|
||||
for prez in req.prezentari:
|
||||
content = prez.model_dump()
|
||||
# T9/OV-2: canonicalize_row inaintea build_key (odometru strip ".0", VIN upper).
|
||||
# canonicalize_row inaintea build_key (odometru strip ".0", VIN upper).
|
||||
# build_key aplica account_or_default(account_id) inainte de hash:
|
||||
# None si 1 colapseaza la aceeasi cheie (canal API + canal import).
|
||||
canon = canonicalize_row(content)
|
||||
key = build_key(account_id, canon)
|
||||
# Aplica normalizarea si in content (odometru canonicalizat inainte de validare, §3.4bis)
|
||||
# Aplica normalizarea si in content (odometru canonicalizat inainte de validare)
|
||||
content.update({
|
||||
"vin": canon["vin"],
|
||||
"nr_inmatriculare": canon["nr_inmatriculare"],
|
||||
@@ -187,7 +225,7 @@ def create_prezentari(
|
||||
(key,),
|
||||
).fetchone()
|
||||
if existing:
|
||||
# US-012: un rand `error` (ex. creds RAR gresite) NU mai blocheaza tacit
|
||||
# Un rand `error` (ex. creds RAR gresite) NU mai blocheaza tacit
|
||||
# retrimiterea aceluiasi continut. Il RE-ACTIVAM (re-clasificam + actualizam
|
||||
# creds + reset), printr-un UPDATE compare-and-swap pe status='error'.
|
||||
if existing["status"] == "error":
|
||||
@@ -205,17 +243,16 @@ def create_prezentari(
|
||||
cl["rar_error"], creds_enc, existing["id"]),
|
||||
)
|
||||
if cur.rowcount == 1:
|
||||
# Creds noi se propaga si in canalul durabil (accounts.rar_creds_enc,
|
||||
# decizie #17) — ambele canale converg pe parola corectata.
|
||||
# Creds noi se propaga si in canalul durabil (accounts.rar_creds_enc)
|
||||
# — ambele canale converg pe parola corectata.
|
||||
if req.rar_credentials is not None:
|
||||
conn.execute(
|
||||
"UPDATE accounts SET rar_creds_enc=? WHERE id=?",
|
||||
(encrypt_creds(req.rar_credentials.model_dump()), acct),
|
||||
)
|
||||
# US-010: telemetrie pentru itemii rezolvati prin regula text.
|
||||
_emite_text_rule_hits(conn, acct, existing["id"], cl["resolved"])
|
||||
# Raspuns onest si la reactivare (PRD 5.7): daca re-clasificarea
|
||||
# cade pe needs_data/needs_mapping, expune motivul (nu doar status).
|
||||
# Raspuns onest si la reactivare: daca re-clasificarea cade pe
|
||||
# needs_data/needs_mapping, expune motivul (nu doar status).
|
||||
results.append(_rezultat_enqueue(existing["id"], cl, reactivated=True))
|
||||
continue
|
||||
# Cursa: alt POST/requeue a schimbat starea intre SELECT si UPDATE
|
||||
@@ -234,8 +271,8 @@ def create_prezentari(
|
||||
)
|
||||
continue
|
||||
|
||||
# Helper pur partajat cu dry-run (PRD 5.2): reproduce EXACT clasificarea
|
||||
# (canonicalize + mapare op->cod + validare + auto_send gate).
|
||||
# Helper pur partajat cu dry-run: reproduce EXACT clasificarea
|
||||
# (canonicalize + mapare op->cod + validare; auto_send gate eliminat dupa US-001).
|
||||
cl = _classify_modal(content, mapping, mapping_meta, valid_codes, error_mode, text_rules)
|
||||
if cl["blocked_error"]:
|
||||
# on_unmapped_error=True: respinge fara enqueue (cod necunoscut/nemapat).
|
||||
@@ -247,13 +284,12 @@ def create_prezentari(
|
||||
(key, acct, cl["status"], json.dumps(cl["content"], ensure_ascii=False), cl["rar_error"], creds_enc),
|
||||
)
|
||||
sub_id = int(cur.lastrowid)
|
||||
# US-010: telemetrie pentru itemii rezolvati prin regula text.
|
||||
_emite_text_rule_hits(conn, acct, sub_id, cl["resolved"])
|
||||
# Raspuns onest (PRD 5.7): pe needs_data/needs_mapping expune erori/nemapate/motiv.
|
||||
# Raspuns onest: pe needs_data/needs_mapping expune erori/nemapate/motiv.
|
||||
results.append(_rezultat_enqueue(sub_id, cl))
|
||||
|
||||
# US-004: audit cerere API per cont. Doar metadate (count + distributie status),
|
||||
# NICIUN camp de payload PII integral. Reuse conn (T4 — fara contentie WAL).
|
||||
# Audit cerere API per cont. Doar metadate (count + distributie status),
|
||||
# NICIUN camp de payload PII integral. Reuse conn (fara contentie WAL).
|
||||
dist: dict[str, int] = {}
|
||||
for r in results:
|
||||
if r.reactivated:
|
||||
@@ -284,7 +320,7 @@ def valideaza_prezentari(
|
||||
|
||||
Intoarce pentru fiecare prezentare: verdictul (status_estimat), erorile de
|
||||
continut si codurile nemapate — exact ce ar obtine trimiterea reala pe acelasi
|
||||
payload + aceeasi mapare de cont. rar_credentials ignorat complet (PRD 5.2).
|
||||
payload + aceeasi mapare de cont. rar_credentials ignorat complet.
|
||||
"""
|
||||
acct = account_or_default(account_id)
|
||||
conn = get_connection()
|
||||
@@ -301,7 +337,7 @@ def valideaza_prezentari(
|
||||
res = _classify_modal(content, mapping, mapping_meta, valid_codes, error_mode, text_rules)
|
||||
if res["blocked_error"]:
|
||||
res = {**res, "status": "error"}
|
||||
# US-003: imbogatim fiecare element nemapat cu 3 niveluri COD_NEMAPAT
|
||||
# Imbogatim fiecare element nemapat cu 3 niveluri COD_NEMAPAT
|
||||
nemapate = [
|
||||
{**u, **err_eroare("COD_NEMAPAT", cauza=f"cod {u.get('cod_op_service')} fara mapare RAR")}
|
||||
for u in res["unmapped"]
|
||||
@@ -329,7 +365,7 @@ def list_prezentari(
|
||||
try:
|
||||
scope_sql, scope_params = account_scope_clause(account_id)
|
||||
# payload_json e plaintext (vezi submissions.payload_json); il citim doar ca
|
||||
# sa derivam campurile afisabile prin helper-ul partajat (US-003, DRY), nu il expunem.
|
||||
# sa derivam campurile afisabile prin helper-ul partajat, nu il expunem.
|
||||
cols = (
|
||||
"id, status, id_prezentare, rar_status_code, retry_count, "
|
||||
"created_at, updated_at, payload_json"
|
||||
@@ -357,13 +393,13 @@ def list_prezentari(
|
||||
conn.close()
|
||||
|
||||
|
||||
# Campuri expuse de GET /v1/prezentari/{id} — allowlist explicita (B4).
|
||||
# Exclude: rar_creds_enc, payload_json, idempotency_key, rar_error, sending_since.
|
||||
# Campuri expuse de GET /v1/prezentari/{id} — allowlist explicita.
|
||||
# Exclude: rar_creds_enc, payload_json, idempotency_key, sending_since.
|
||||
_PREZENTARE_FIELDS = frozenset({
|
||||
"id", "status", "id_prezentare", "rar_status_code", "retry_count",
|
||||
"next_attempt_at", "created_at", "updated_at", "account_id",
|
||||
"batch_id", "row_index", "purge_after",
|
||||
# T9: rar_error e SIGUR de expus — contine doar coduri/mesaje de validare RAR si
|
||||
# rar_error e SIGUR de expus — contine doar coduri/mesaje de validare RAR si
|
||||
# erori din catalog (niciodata creds, ex. RAR_CREDS_INVALIDE poarta doar cauza
|
||||
# "credentiale RAR invalide", fara parola). Face recovery-ul observabil prin API.
|
||||
"rar_error",
|
||||
@@ -383,7 +419,7 @@ def get_prezentare(
|
||||
[submission_id] + scope_params,
|
||||
).fetchone()
|
||||
if not row:
|
||||
# B3: acelasi mesaj indiferent daca randul exista dar apartine altui cont
|
||||
# Acelasi mesaj indiferent daca randul exista dar apartine altui cont
|
||||
# sau nu exista deloc — nu confirmam existenta.
|
||||
raise HTTPException(status_code=404, detail="submission inexistent")
|
||||
row_dict = dict(row)
|
||||
@@ -397,11 +433,11 @@ def delete_prezentare(
|
||||
submission_id: int,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
) -> dict:
|
||||
"""Sterge o trimitere blocata a contului cheii API (US-010).
|
||||
"""Sterge o trimitere blocata a contului cheii API.
|
||||
|
||||
Raspuns 200 + body JSON (NU 204 — clienti VFP fac string-parse). Scope evaluat
|
||||
INAINTEA starii (decizie /autoplan #20): cross-account / inexistent -> 404 (acelasi
|
||||
mesaj, B3); own-account `sent`/`sending` -> 409 (conflict de stare).
|
||||
INAINTEA starii: cross-account / inexistent -> 404 (acelasi mesaj);
|
||||
own-account `sent`/`sending` -> 409 (conflict de stare).
|
||||
"""
|
||||
conn = get_connection()
|
||||
try:
|
||||
@@ -424,10 +460,10 @@ def repune_prezentare(
|
||||
submission_id: int,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
) -> dict:
|
||||
"""Re-pune in coada o trimitere blocata a contului cheii API (US-010).
|
||||
"""Re-pune in coada o trimitere blocata a contului cheii API.
|
||||
|
||||
`error -> queued` (peste helper US-009), re-ruleaza classify. Acelasi oracol de
|
||||
scope/stare ca DELETE (404 cross-account/inexistent, 409 sent/sending).
|
||||
`error -> queued`, re-ruleaza classify. Acelasi oracol de scope/stare ca DELETE
|
||||
(404 cross-account/inexistent, 409 sent/sending).
|
||||
"""
|
||||
conn = get_connection()
|
||||
try:
|
||||
@@ -478,8 +514,7 @@ def _audit_rows(conn, date_from: str | None, date_to: str | None, status: str, a
|
||||
"""Randuri audit filtrate pe cont + data(updated_at) in [from, to].
|
||||
|
||||
account_id = contul cheii API (scope obligatoriu — PII in CSV). Randuri cu
|
||||
account_id IS NULL apartin contului 1 (legacy/OV-2). payload_json e text in
|
||||
schelet; b64_image NU intra in CSV.
|
||||
account_id IS NULL apartin contului 1. b64_image NU intra in CSV.
|
||||
"""
|
||||
scope_sql, scope_params = account_scope_clause(account_id)
|
||||
sql = (
|
||||
@@ -514,7 +549,7 @@ def _audit_rows(conn, date_from: str | None, date_to: str | None, status: str, a
|
||||
"submission_id": r["id"],
|
||||
"status": r["status"],
|
||||
"id_prezentare": r["id_prezentare"] or "",
|
||||
# NULL→cont 1 (OV-2): coloana reflecta invariantul de scope, nu "" ambiguu.
|
||||
# NULL→cont 1: coloana reflecta invariantul de scope, nu "" ambiguu.
|
||||
"account_id": account_or_default(r["account_id"]),
|
||||
"vin": p.get("vin") or "",
|
||||
"nr_inmatriculare": p.get("nr_inmatriculare") or "",
|
||||
@@ -539,7 +574,7 @@ def audit_export(
|
||||
|
||||
pe data(updated_at). `status` implicit `sent` (ce a ajuns efectiv la RAR);
|
||||
`status=all` exporta toata coada contului. Leaga retinerea 90 zile prin coloana
|
||||
`purge_after` (plan.md sect. 4 + 8). b64_image nu se exporta.
|
||||
`purge_after`. b64_image nu se exporta.
|
||||
"""
|
||||
conn = get_connection()
|
||||
try:
|
||||
@@ -568,7 +603,7 @@ def get_mapari(
|
||||
"""Maparile operatie->cod ale contului curent.
|
||||
|
||||
Parametrul `account_id` din query e pastrat pentru compatibilitate, dar contul
|
||||
efectiv vine MEREU din cheia API (TD-3.2). Daca e prezent si difera -> 400.
|
||||
efectiv vine MEREU din cheia API. Daca e prezent si difera -> 400.
|
||||
"""
|
||||
if account_id is not None and account_id != key_account:
|
||||
raise HTTPException(
|
||||
@@ -635,7 +670,7 @@ def create_mapare(
|
||||
|
||||
|
||||
class RarCredsIn(BaseModel):
|
||||
"""Creds RAR durabile per-cont (D4). Stocate criptate (Fernet) in accounts.rar_creds_enc."""
|
||||
"""Creds RAR durabile per-cont. Stocate criptate (Fernet) in accounts.rar_creds_enc."""
|
||||
|
||||
email: str = Field(..., min_length=1)
|
||||
password: str = Field(..., min_length=1, repr=False)
|
||||
@@ -646,7 +681,7 @@ def set_rar_creds(
|
||||
req: RarCredsIn,
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
) -> dict:
|
||||
"""Seteaza creds RAR durabile per-cont (D4/T1).
|
||||
"""Seteaza creds RAR durabile per-cont.
|
||||
|
||||
Criptate Fernet in accounts.rar_creds_enc. Worker-ul le foloseste ca fallback
|
||||
cand submission-ul nu mai are creds (canal web fara re-pusher, restart worker).
|
||||
|
||||
63
app/auth.py
63
app/auth.py
@@ -18,8 +18,9 @@ from __future__ import annotations
|
||||
import hashlib
|
||||
import secrets
|
||||
import sqlite3
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from fastapi import Header, HTTPException, Request
|
||||
from fastapi import Depends, Header, HTTPException, Request
|
||||
|
||||
from .config import get_settings
|
||||
from .db import get_connection
|
||||
@@ -112,7 +113,7 @@ def _extract_key(x_api_key: str | None, authorization: str | None) -> str | None
|
||||
|
||||
|
||||
def _log_auth_esuat(request: Request | None, plaintext: str | None, motiv: str) -> None:
|
||||
"""Eveniment de jurnal pentru un esec de auth (US-004): IP + prefix cheie, NU cheia.
|
||||
"""Eveniment de jurnal pentru un esec de auth: IP + prefix cheie, NU cheia.
|
||||
|
||||
Best-effort (log_event inghite erorile). Import local: evita cuplarea la import-time
|
||||
(observ -> db; auth -> db) si pastreaza auth.py importabil din CLI fara efecte.
|
||||
@@ -142,7 +143,7 @@ def resolve_account_id(
|
||||
- cheie invalida (prezenta) -> 401 (mereu, indiferent de flag)
|
||||
- fara cheie + flag off -> cont implicit (id=1), back-compat
|
||||
- fara cheie + flag on -> 401
|
||||
Esecurile de auth (401) emit `api_auth_esuat` cu IP + prefix cheie (US-004).
|
||||
Esecurile de auth (401) emit `api_auth_esuat` cu IP + prefix cheie.
|
||||
"""
|
||||
settings = get_settings()
|
||||
plaintext = _extract_key(x_api_key, authorization)
|
||||
@@ -162,3 +163,59 @@ def resolve_account_id(
|
||||
_log_auth_esuat(request, plaintext, "cheie API invalida sau revocata")
|
||||
raise HTTPException(status_code=401, detail="cheie API invalida sau revocata")
|
||||
return account_id
|
||||
|
||||
|
||||
def require_api_access(
|
||||
account_id: int = Depends(resolve_account_id),
|
||||
) -> int:
|
||||
"""Dependency FastAPI (T4, PRD 5.17): verifica ca tier-ul efectiv permite accesul la API.
|
||||
|
||||
Reguli:
|
||||
- enforce_plans=False (kill-switch): sare verificarea.
|
||||
- dev id=1 cu require_api_key=False: bypass (dogfooding, testele existente nu pica).
|
||||
- Pro/Premium sau trial Pro activ: permit.
|
||||
- Free/Standard fara trial: 403 PLAN_FARA_API cu eroare 3 niveluri.
|
||||
|
||||
Refoloseste resolve_account_id (account_id deja rezolvat din cheie API).
|
||||
Se ataseaza ca Depends() pe rutele de ingestie API (POST /v1/prezentari,
|
||||
POST /v1/import, POST /v1/import/{id}/commit). valideaza + nomenclator raman libere.
|
||||
"""
|
||||
from .plans import PLANS, effective_tier
|
||||
from .errors import eroare as _eroare
|
||||
|
||||
settings = get_settings()
|
||||
# Kill-switch operare: sare toate gate-urile de plan.
|
||||
if not settings.enforce_plans:
|
||||
return account_id
|
||||
# Bypass pentru contul implicit dev (id=1) in modul fara cheie API obligatorie.
|
||||
# In prod (require_api_key=True), id=1 nu are bypass implicit (cheie = obligatorie).
|
||||
if not settings.require_api_key and account_id == DEFAULT_ACCOUNT_ID:
|
||||
return account_id
|
||||
|
||||
conn = get_connection()
|
||||
try:
|
||||
row = conn.execute(
|
||||
"SELECT tier, trial_until FROM accounts WHERE id=?", (account_id,)
|
||||
).fetchone()
|
||||
finally:
|
||||
conn.close()
|
||||
|
||||
now = datetime.now(timezone.utc)
|
||||
et = effective_tier(row, now)
|
||||
if not PLANS[et].get("api_access"):
|
||||
from .observ import log_event
|
||||
log_event(
|
||||
"plan_api_refuzat",
|
||||
account_id=account_id,
|
||||
nivel="WARNING",
|
||||
mesaj=f"Acces API refuzat: tier efectiv={et}",
|
||||
context={"tier_efectiv": et},
|
||||
)
|
||||
raise HTTPException(
|
||||
status_code=403,
|
||||
detail=_eroare(
|
||||
"PLAN_FARA_API",
|
||||
cauza=f"Tier efectiv: {et}. API disponibil pe Pro/Premium.",
|
||||
),
|
||||
)
|
||||
return account_id
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
"""Configurare gateway. Env vars (prefix AUTOPASS_) + valori implicite.
|
||||
|
||||
NU stocheaza parole RAR. Credentialele RAR vin per-cerere de la ROAAUTO
|
||||
(vezi plan.md sect. 5). Helper-ul `load_test_credentials` citeste blocul
|
||||
<test> din settings.xml DOAR pentru dev local / probe pe mediul de test.
|
||||
NU stocheaza parole RAR. Credentialele RAR vin per-cerere de la ROAAUTO.
|
||||
Helper-ul `load_test_credentials` citeste blocul <test> din settings.xml DOAR
|
||||
pentru dev local / probe pe mediul de test.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -22,22 +22,21 @@ class Settings(BaseSettings):
|
||||
# --- Bază de date ---
|
||||
db_path: Path = ROOT / "data" / "autopass.db"
|
||||
|
||||
# --- Observabilitate / jurnal aplicatie (PRD 5.6) ---
|
||||
# --- Observabilitate / jurnal aplicatie ---
|
||||
# Nivel minim al evenimentelor scrise in app_events + log text. Sub el, evenimentul
|
||||
# e ignorat (best-effort). DEBUG|INFO|WARNING|ERROR|CRITICAL.
|
||||
log_level: str = "INFO"
|
||||
# Retentie jurnal (app_events) — aliniat cu submissions/import_batches (decizie §5).
|
||||
log_retention_days: int = 90
|
||||
# Director pentru log-ul text rotativ (RotatingFileHandler in aplicatie, decizie §5).
|
||||
# Director pentru log-ul text rotativ (RotatingFileHandler in aplicatie).
|
||||
# Fisier per-proces (app-api.log / app-worker.log) — rotatia nu e multiproces-safe.
|
||||
log_dir: Path = ROOT / ".run"
|
||||
log_file_max_bytes: int = 5_000_000
|
||||
log_file_backup_count: int = 5
|
||||
# Retentie randuri blocate (error/needs_data/needs_mapping). Mai scurt decat 90z
|
||||
# ale `sent` — un blocat n-are valoare de audit (decizie §5).
|
||||
# ale `sent` — un blocat n-are valoare de audit.
|
||||
blocked_retention_days: int = 30
|
||||
|
||||
# --- Securitate (CORE) ---
|
||||
# --- Securitate ---
|
||||
# Enforcement auth API-key pe /v1/* protejat. False (dev/test): fara cheie ->
|
||||
# cont implicit id=1. True (prod): fara cheie valida -> 401. O cheie PREZENTA
|
||||
# dar invalida da 401 indiferent de flag.
|
||||
@@ -49,29 +48,33 @@ class Settings(BaseSettings):
|
||||
# python -c "from cryptography.fernet import Fernet; print(Fernet.generate_key().decode())"
|
||||
creds_key: str | None = None
|
||||
|
||||
# --- Sesiuni web (US-002, PRD 3.3) ---
|
||||
# --- Sesiuni web ---
|
||||
# Secret semnat cookie sesiune. None -> efemer la fiecare restart (dev ok;
|
||||
# in prod seteaza persistent ca si creds_key, altfel cookieurile se invalideaza
|
||||
# la restart). Genereaza: python -c "import secrets; print(secrets.token_hex(32))"
|
||||
session_secret: str | None = None
|
||||
# True (IMPLICIT, sigur pentru prod): rutele web fara sesiune -> redirect /login;
|
||||
# CSRF enforce. Pentru dev rapid pe contul implicit id=1 (back-compat C12/§5 Q5),
|
||||
# CSRF enforce. Pentru dev rapid pe contul implicit id=1,
|
||||
# seteaza explicit AUTOPASS_WEB_AUTH_REQUIRED=false.
|
||||
web_auth_required: bool = True
|
||||
# True (prod, in spatele Cloudflare Tunnel TLS): cookie cu Secure flag (C4).
|
||||
# True (prod, in spatele Cloudflare Tunnel TLS): cookie cu Secure flag.
|
||||
# False (dev): cookie fara Secure, functioneaza pe HTTP.
|
||||
session_https_only: bool = False
|
||||
|
||||
# --- Notificare email admin la signup (US-012, PRD 3.3b) ---
|
||||
# Nesetat (smtp_host None) -> notificarea e DEGRADATA (doar log SIGNUP);
|
||||
# follow-up cand exista SMTP real configurat in .env.
|
||||
# --- Contact suport (US-001, PRD 5.12) ---
|
||||
# Email/canal de contact afisat in mesaje catre utilizatori (ex. CUI duplicat la signup).
|
||||
# Nesetat -> fallback la formularea generica fara canal concret.
|
||||
support_email: str | None = None
|
||||
|
||||
# --- Notificare email admin la signup ---
|
||||
# Nesetat (smtp_host None) -> notificarea e DEGRADATA (doar log SIGNUP).
|
||||
smtp_host: str | None = None
|
||||
smtp_port: int = 587
|
||||
smtp_user: str | None = None
|
||||
smtp_password: str | None = None
|
||||
smtp_from: str | None = None
|
||||
|
||||
# --- Rate-limit signup + login (US-009, PRD 3.3 C5) ---
|
||||
# --- Rate-limit signup + login ---
|
||||
# Max cereri POST /signup per IP in fereastra de timp (in-proces, fara dependinta noua).
|
||||
signup_rate_max: int = 5
|
||||
signup_rate_window_s: int = 3600
|
||||
@@ -83,25 +86,52 @@ class Settings(BaseSettings):
|
||||
rar_base_url_test: str = "https://apps.rarom.ro/test-rar-autopass"
|
||||
rar_base_url_prod: str = "https://apps.rarom.ro/rar-autopass"
|
||||
|
||||
# WAF-ul RAR da 403 fara User-Agent de browser (confirmat live, vezi
|
||||
# docs/api-rar-contract.md). Toate apelurile httpx il trimit.
|
||||
# WAF-ul RAR da 403 fara User-Agent de browser. Toate apelurile httpx il trimit.
|
||||
http_user_agent: str = "Mozilla/5.0"
|
||||
http_timeout_s: float = 30.0
|
||||
|
||||
# --- Worker ---
|
||||
worker_poll_interval_s: float = 5.0
|
||||
worker_heartbeat_stale_s: int = 30 # /healthz considera worker-ul mort peste atat
|
||||
# In schelet send-ul e DEZACTIVAT (nu trimite la RAR). Activeaza-l explicit
|
||||
# pentru proba end-to-end. Reconcilierea/retry-ul complet = T2.
|
||||
# Send DEZACTIVAT implicit (nu trimite la RAR). Activeaza-l explicit pentru
|
||||
# proba end-to-end.
|
||||
worker_send_enabled: bool = False
|
||||
# Dev: foloseste creds <test> din settings.xml pt login worker. In productie
|
||||
# creds vin per-cerere de la ROAAUTO (T2) — lasa False.
|
||||
# creds vin per-cerere de la ROAAUTO — lasa False.
|
||||
worker_use_test_creds: bool = False
|
||||
# T2 — recuperare orfane + retry/backoff:
|
||||
# Keepalive RAR: cand coada e goala, worker-ul face un login de proba la fiecare
|
||||
# atata timp ca sa pastreze last_rar_login_ok proaspat (sub pragul de 30h al
|
||||
# dashboard-ului) — altfel banner-ul "RAR inaccesibil" apare fals doar din lipsa
|
||||
# de trafic. 0 = dezactivat. Implicit o data pe zi (24h < 30h, margine de 6h).
|
||||
worker_rar_keepalive_interval_s: int = 86400
|
||||
worker_sending_lease_s: int = 120 # rand 'sending' mai vechi de atat = orfan (worker mort mid-POST)
|
||||
worker_retry_base_s: int = 5 # backoff = base * 2^retry (plafonat la max)
|
||||
worker_retry_max_s: int = 300
|
||||
worker_max_retries: int = 8 # peste atat -> error + banner (pana persistenta)
|
||||
worker_max_retries: int = 8 # peste atat -> error + banner
|
||||
|
||||
# --- Planuri de cont (PRD 5.17) ---
|
||||
# Enforcement DUR al limitelor de plan (volum + acces API). True (implicit) = activ.
|
||||
# False = kill-switch de operare: sare toate gate-urile de plan (util pentru debugging
|
||||
# sau rollback rapid fara revert de cod). Enforcement DUR e activ implicit de la deploy
|
||||
# (decizie user 2026-06-28, decizia #22 autoplan): nu exista conturi legacy, produs in TESTE.
|
||||
enforce_plans: bool = True
|
||||
|
||||
# --- Embeddings (sugestie mapare, Stratul 2 PRD 5.14) ---
|
||||
# ACTIVAT implicit: editorul de mapari ofera sugestii semantice (model fastembed/ONNX).
|
||||
# Cost: prima folosire lazy-load-eaza modelul (~230MB pe disc) sincron in thread-ul de
|
||||
# cerere -> prima cerere /mapari poate dura 30-120s pana modelul intra in memorie; cererile
|
||||
# urmatoare sunt instant. SUGGESTION-ONLY: nu intra in resolve_prestatii (nu auto-trimite).
|
||||
# Pune-l pe False (start.sh/Docker/.env: AUTOPASS_EMBEDDINGS_ENABLED=false) cand vrei
|
||||
# /mapari instant la prima cerere sau suita de teste rapida (cade pe GOLD/SILVER+fuzzy).
|
||||
embeddings_enabled: bool = True
|
||||
|
||||
# --- Seed corpus operatii etichetate (SILVER, PRD 5.18 US-004) ---
|
||||
# ACTIVAT implicit: la init_db, populeaza mapping_suggestions din artefactul comis
|
||||
# `app/data/operatii-etichetate.json` (INSERT OR IGNORE). Asa SILVER nu mai e gol in
|
||||
# productie -> sugestii exact-match + corpus k-NN reale. SUGGESTION-ONLY.
|
||||
# Pune-l pe False (AUTOPASS_SEED_OPERATII_ENABLED=false) cand vrei SILVER gol —
|
||||
# conftest il dezactiveaza global, testele care-l vor il pornesc punctual.
|
||||
seed_operatii_enabled: bool = True
|
||||
|
||||
@property
|
||||
def rar_base_url(self) -> str:
|
||||
|
||||
137450
app/data/operatii-etichetate.json
Normal file
137450
app/data/operatii-etichetate.json
Normal file
File diff suppressed because it is too large
Load Diff
166
app/db.py
166
app/db.py
@@ -37,6 +37,22 @@ def init_db() -> None:
|
||||
from .mapping import seed_nomenclator_if_empty
|
||||
|
||||
seed_nomenclator_if_empty(conn)
|
||||
# Seed corpus operatii etichetate -> mapping_suggestions (SILVER, PRD 5.18 US-004).
|
||||
# Gated: OFF in teste (conftest), ON in productie. INSERT OR IGNORE -> idempotent.
|
||||
# DOAR daca mapping_suggestions e gol: seedul are ~17k randuri; re-rularea lui pe
|
||||
# FIECARE boot (API + worker concurent) tinea write-lock-ul indelung -> al doilea
|
||||
# proces primea "database is locked" la pornire. Guard "_if_empty" (ca nomenclatorul)
|
||||
# -> boot rapid cand e deja seeded. Re-seed dupa actualizarea fisierului = manual
|
||||
# (goleste tabela), consistent cu semantica v1 ignore-not-upsert a seederului.
|
||||
if get_settings().seed_operatii_enabled:
|
||||
already = conn.execute(
|
||||
"SELECT 1 FROM mapping_suggestions LIMIT 1"
|
||||
).fetchone()
|
||||
if not already:
|
||||
from .operatii_seed import seed_operatii_etichetate
|
||||
|
||||
seed_operatii_etichetate(conn)
|
||||
conn.commit()
|
||||
finally:
|
||||
conn.close()
|
||||
|
||||
@@ -55,19 +71,34 @@ def _migrate(conn: sqlite3.Connection) -> None:
|
||||
conn.execute("ALTER TABLE submissions ADD COLUMN batch_id INTEGER")
|
||||
if "row_index" not in sub_cols:
|
||||
conn.execute("ALTER TABLE submissions ADD COLUMN row_index INTEGER")
|
||||
if "rar_env" not in sub_cols:
|
||||
# PRD 5.20 US-001. Mediul RAR tinta pe submission. Pe DB existent NU lasam
|
||||
# randurile pe DEFAULT 'test': un rand prod pre-migrare etichetat 'test' ar fi
|
||||
# reconciliat de worker (US-006) contra endpoint TEST -> no-match -> re-send prod
|
||||
# = DUPLICAT REAL IREVERSIBIL. Backfill din AUTOPASS_RAR_ENV global (ancora de
|
||||
# migrare) + recompute idempotency_key env-aware. Ruleaza O SINGURA DATA (in
|
||||
# blocul de adaugare a coloanei); pe DB fresh coloana vine din schema.sql (fara rows).
|
||||
conn.execute(
|
||||
"ALTER TABLE submissions ADD COLUMN rar_env TEXT NOT NULL DEFAULT 'test' "
|
||||
"CHECK (rar_env IN ('test', 'prod'))"
|
||||
)
|
||||
_backfill_submissions_rar_env(conn)
|
||||
|
||||
# Coloane accounts
|
||||
acc_cols = {r["name"] for r in conn.execute("PRAGMA table_info(accounts)").fetchall()}
|
||||
if "rar_creds_enc" not in acc_cols:
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN rar_creds_enc TEXT")
|
||||
acc_cols.add("rar_creds_enc")
|
||||
# Medii RAR per cont (PRD 5.20 US-001): activare + slot creds + default, per mediu.
|
||||
_migrate_accounts_medii(conn, acc_cols)
|
||||
if "active" not in acc_cols:
|
||||
# Conturi existente raman active (default 1). Lifecycle consumat de 3.3.
|
||||
# Conturi existente raman active (default 1).
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN active INTEGER NOT NULL DEFAULT 1")
|
||||
acc_cols.add("active")
|
||||
if "status" not in acc_cols:
|
||||
# Stare de ciclu de viata (5.5). Defensiv idempotent (ca is_admin in 3.3b).
|
||||
# Default 'active' (trece CHECK pe randurile existente), apoi derivam din `active`:
|
||||
# active=0 -> 'pending'. Invariant: active=1 <=> status='active'.
|
||||
# Stare de ciclu de viata. Default 'active' (trece CHECK pe randurile existente),
|
||||
# apoi derivam din `active`: active=0 -> 'pending'.
|
||||
# Invariant: active=1 <=> status='active'.
|
||||
conn.execute(
|
||||
"ALTER TABLE accounts ADD COLUMN status TEXT NOT NULL DEFAULT 'active' "
|
||||
"CHECK (status IN ('pending','active','blocked','archived','deleted'))"
|
||||
@@ -81,6 +112,26 @@ def _migrate(conn: sqlite3.Connection) -> None:
|
||||
"ALTER TABLE accounts ADD COLUMN on_unmapped_error_default INTEGER NOT NULL DEFAULT 0 "
|
||||
"CHECK (on_unmapped_error_default IN (0, 1))"
|
||||
)
|
||||
if "email" not in acc_cols:
|
||||
# Email canonic de contact al firmei (US-001, PRD 5.12). Nullable pt. conturi legacy.
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN email TEXT")
|
||||
if "tier" not in acc_cols:
|
||||
# Plan de cont (US-001, PRD 5.17). Legacy -> 'free' fara trial (enforcement DUR la deploy).
|
||||
conn.execute(
|
||||
"ALTER TABLE accounts ADD COLUMN tier TEXT NOT NULL DEFAULT 'free' "
|
||||
"CHECK (tier IN ('free','standard','pro','premium'))"
|
||||
)
|
||||
if "trial_until" not in acc_cols:
|
||||
# Trial Pro activ daca != NULL si > now. Nullable (NULL = fara trial).
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN trial_until TEXT")
|
||||
if "requested_plan" not in acc_cols:
|
||||
# Planul cerut la signup (integrare plati). NU acorda drepturi; `tier` ramane sursa
|
||||
# de adevar pt API/volum. Nullable. ALTER nu poate adauga CHECK pe coloana noua in
|
||||
# SQLite -> validarea valorilor se face in cod (signup, fata de VALID_TIERS).
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN requested_plan TEXT")
|
||||
if "consent_at" not in acc_cols:
|
||||
# Marca temporala consimtamant Termeni+GDPR (proba). Nullable (NULL = CLI/legacy).
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN consent_at TEXT")
|
||||
# Unicitate CUI (un CUI = un cont); NULL distinct nativ -> conturi fara CUI multiplu.
|
||||
conn.execute(
|
||||
"CREATE UNIQUE INDEX IF NOT EXISTS ux_accounts_cui ON accounts(cui) WHERE cui IS NOT NULL"
|
||||
@@ -97,9 +148,7 @@ def _migrate(conn: sqlite3.Connection) -> None:
|
||||
if "email_verified" not in user_cols:
|
||||
conn.execute("ALTER TABLE users ADD COLUMN email_verified INTEGER NOT NULL DEFAULT 0")
|
||||
|
||||
# Coloana import_rows.override_json (3.6, Approach B): patch canonic editat in
|
||||
# preview, criptat Fernet. Defensiv idempotent (ca is_admin in 3.3b) — DB create
|
||||
# inainte de 3.6 nu au coloana.
|
||||
# Coloana import_rows.override_json: patch canonic editat in preview, criptat Fernet.
|
||||
irows_tbl = conn.execute(
|
||||
"SELECT name FROM sqlite_master WHERE type='table' AND name='import_rows'"
|
||||
).fetchone()
|
||||
@@ -107,6 +156,12 @@ def _migrate(conn: sqlite3.Connection) -> None:
|
||||
irows_cols = {r["name"] for r in conn.execute("PRAGMA table_info(import_rows)").fetchall()}
|
||||
if "override_json" not in irows_cols:
|
||||
conn.execute("ALTER TABLE import_rows ADD COLUMN override_json TEXT")
|
||||
if "reviewed" not in irows_cols:
|
||||
# Marcaj confirmare umana (US-007, PRD 5.12). NU intra in payload/idempotenta.
|
||||
# NOT NULL DEFAULT 0: valoare clara (0=neconfirmat), fara ambiguitate NULL vs 0.
|
||||
conn.execute(
|
||||
"ALTER TABLE import_rows ADD COLUMN reviewed INTEGER NOT NULL DEFAULT 0"
|
||||
)
|
||||
|
||||
# Index batch_id pe submissions (poate lipsi pe DB veche)
|
||||
existing_idx = {r["name"] for r in conn.execute(
|
||||
@@ -124,6 +179,101 @@ def _migrate(conn: sqlite3.Connection) -> None:
|
||||
)
|
||||
|
||||
|
||||
def _migrate_accounts_medii(conn: sqlite3.Connection, acc_cols: set[str]) -> None:
|
||||
"""PRD 5.20 US-001: coloane medii RAR per cont + backfill din ancora globala.
|
||||
|
||||
Adauga (idempotent): rar_test_enabled/rar_prod_enabled (bife activare),
|
||||
rar_creds_test_enc/rar_creds_prod_enc (sloturi creds), rar_env_default.
|
||||
|
||||
Backfill (O SINGURA DATA, cand coloanele tocmai au fost adaugate pe DB existent):
|
||||
creds-ul legacy `rar_creds_enc` apartine mediului `AUTOPASS_RAR_ENV` global de la
|
||||
momentul migrarii (ancora) — il copiem in slotul acelui mediu, activam DOAR acel
|
||||
mediu (celalalt dezactivat) si fixam default-ul pe el. Conturile fara creds raman
|
||||
pe default-urile coloanei (prod on / test off). Migrarea NU presupune env-ul; se
|
||||
bazeaza pe ancora globala, exact cum opera contul inainte de 5.20.
|
||||
"""
|
||||
newly_added = "rar_env_default" not in acc_cols
|
||||
if "rar_test_enabled" not in acc_cols:
|
||||
conn.execute(
|
||||
"ALTER TABLE accounts ADD COLUMN rar_test_enabled INTEGER NOT NULL DEFAULT 0 "
|
||||
"CHECK (rar_test_enabled IN (0, 1))"
|
||||
)
|
||||
if "rar_prod_enabled" not in acc_cols:
|
||||
conn.execute(
|
||||
"ALTER TABLE accounts ADD COLUMN rar_prod_enabled INTEGER NOT NULL DEFAULT 1 "
|
||||
"CHECK (rar_prod_enabled IN (0, 1))"
|
||||
)
|
||||
if "rar_creds_test_enc" not in acc_cols:
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN rar_creds_test_enc TEXT")
|
||||
if "rar_creds_prod_enc" not in acc_cols:
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN rar_creds_prod_enc TEXT")
|
||||
if "rar_env_default" not in acc_cols:
|
||||
# ALTER nu poate adauga CHECK pe coloana noua in SQLite -> validarea ('test'/'prod')
|
||||
# se face in cod (rar_env.py / rutele de cont). DEFAULT 'prod' (cont client nou).
|
||||
conn.execute("ALTER TABLE accounts ADD COLUMN rar_env_default TEXT NOT NULL DEFAULT 'prod'")
|
||||
|
||||
if not newly_added:
|
||||
return # coloanele existau deja -> backfill-ul a rulat la o pornire anterioara
|
||||
|
||||
# Are coloana legacy rar_creds_enc randuri de migrat? (Pe DB foarte nou, e absenta.)
|
||||
if "rar_creds_enc" not in acc_cols:
|
||||
return
|
||||
env = get_settings().rar_env if get_settings().rar_env in ("test", "prod") else "test"
|
||||
other = "prod" if env == "test" else "test"
|
||||
slot = f"rar_creds_{env}_enc"
|
||||
conn.execute(
|
||||
f"UPDATE accounts SET {slot} = rar_creds_enc, "
|
||||
f"rar_{env}_enabled = 1, rar_{other}_enabled = 0, rar_env_default = ? "
|
||||
f"WHERE rar_creds_enc IS NOT NULL AND TRIM(rar_creds_enc) <> '' AND {slot} IS NULL",
|
||||
(env,),
|
||||
)
|
||||
|
||||
|
||||
def _backfill_submissions_rar_env(conn: sqlite3.Connection) -> None:
|
||||
"""PRD 5.20 US-001 (AUTO-FIX G + E4/3): backfill rar_env + recompute idempotency_key.
|
||||
|
||||
Ruleaza O SINGURA DATA, imediat dupa ce coloana `submissions.rar_env` a fost adaugata
|
||||
pe un DB existent. Toate randurile pre-migrare au fost trimise (sau urmeaza) catre
|
||||
mediul `AUTOPASS_RAR_ENV` global — le etichetam cu acel env (NU DEFAULT 'test'), altfel
|
||||
reconcilierea worker-ului ar lovi endpoint-ul gresit -> duplicat ireversibil.
|
||||
|
||||
Recompute `idempotency_key` la forma env-aware (`build_key(account_id, canon, rar_env)`):
|
||||
altfel un re-POST al unui rand legacy (cheie env-less) ar rata randul existent ->
|
||||
duplicat. Recompute-ul e consistent (acelasi env pe toate randurile pre-migrare) deci
|
||||
nu poate crea coliziuni intre randuri care erau deja distincte.
|
||||
"""
|
||||
import json as _json
|
||||
|
||||
from .idempotency import build_key, canonicalize_row
|
||||
|
||||
env = get_settings().rar_env if get_settings().rar_env in ("test", "prod") else "test"
|
||||
conn.execute("UPDATE submissions SET rar_env = ?", (env,))
|
||||
|
||||
rows = conn.execute(
|
||||
"SELECT id, account_id, idempotency_key, payload_json FROM submissions"
|
||||
).fetchall()
|
||||
for r in rows:
|
||||
try:
|
||||
content = _json.loads(r["payload_json"])
|
||||
except (ValueError, TypeError):
|
||||
continue
|
||||
canon = canonicalize_row(content)
|
||||
# Pastreaza prestatiile rezolvate (cod_prestatie/cod_op_service) pentru _op_identity.
|
||||
canon["prestatii"] = content.get("prestatii") or []
|
||||
new_key = build_key(r["account_id"], canon, env)
|
||||
if new_key == r["idempotency_key"]:
|
||||
continue
|
||||
try:
|
||||
conn.execute(
|
||||
"UPDATE submissions SET idempotency_key = ? WHERE id = ?",
|
||||
(new_key, r["id"]),
|
||||
)
|
||||
except sqlite3.IntegrityError:
|
||||
# Coliziune improbabila pe UNIQUE(idempotency_key): lasa cheia veche (no-op),
|
||||
# randul ramane gasibil prin dual-lookup legacy.
|
||||
continue
|
||||
|
||||
|
||||
def _now_iso() -> str:
|
||||
return datetime.now(timezone.utc).isoformat(timespec="seconds")
|
||||
|
||||
@@ -151,7 +301,7 @@ def queue_depth(conn: sqlite3.Connection) -> int:
|
||||
return int(row["n"]) if row else 0
|
||||
|
||||
|
||||
# --- Jurnal de aplicatie (app_events, PRD 5.6 US-003) ---
|
||||
# --- Jurnal de aplicatie (app_events) ---
|
||||
|
||||
def insert_app_event(
|
||||
conn: sqlite3.Connection,
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
"""Helper notificare email admin la signup (US-012, PRD 3.3b).
|
||||
"""Helper notificare email admin la signup.
|
||||
|
||||
Livrare DEGRADATA: daca smtp_host nu e configurat, functia e no-op (log doar).
|
||||
Orice eroare SMTP e prinsa si logata — signup-ul NU e blocat niciodata.
|
||||
|
||||
249
app/embeddings.py
Normal file
249
app/embeddings.py
Normal file
@@ -0,0 +1,249 @@
|
||||
"""Modul embedding in-proces pentru sugestie cod RAR -- L14-S4.
|
||||
|
||||
Design (PRD 5.14, Decision #16/#16b):
|
||||
- Model multilingv via fastembed/ONNX (~230MB pe disc, quantizat, fara torch)
|
||||
- Lazy load la prima folosire, NU la import si NU pe /healthz
|
||||
- Worker NU incarca modelul (API-only)
|
||||
- Degradare gratioasa: daca modelul nu se incarca -> is_available()=False,
|
||||
suggest_nearest() -> [] fara exceptie, ingestia NU e blocata
|
||||
- Embeddings = DOAR sugestie (nu intra in lantul de enqueue/resolve_prestatii)
|
||||
- NU apelat din resolve_prestatii/load_mapping (wiring vine in L14-S6 DUPA 5.15)
|
||||
|
||||
API public (nivel modul):
|
||||
index_corpus(items) -> None
|
||||
suggest_nearest(text, top_k) -> [{cod, is_nul, similaritate}]
|
||||
is_available() -> bool
|
||||
|
||||
Clase (pentru teste / injectare backend):
|
||||
EmbeddingEngine(backend) -- motor testabil cu backend injectabil
|
||||
FastEmbedBackend() -- backend real fastembed/ONNX
|
||||
"""
|
||||
from __future__ import annotations
|
||||
|
||||
import logging
|
||||
import math
|
||||
from typing import Protocol, runtime_checkable
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
# Modelul ales: paraphrase-multilingual-MiniLM-L12-v2
|
||||
# ~230MB pe disc (ONNX quantizat), 384 dim, multilingv (ro/en/etc.), suportat de
|
||||
# fastembed, fara torch. (Estimarea initiala din PRD de ~50MB a fost gresita.)
|
||||
FASTEMBED_MODEL = "sentence-transformers/paraphrase-multilingual-MiniLM-L12-v2"
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Protocol backend (mockabil in teste) #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
@runtime_checkable
|
||||
class EmbeddingBackend(Protocol):
|
||||
"""Interfata minimala pentru un backend de embedding."""
|
||||
|
||||
def embed(self, texts: list[str]) -> list[list[float]]:
|
||||
"""Intoarce o lista de vectori (cate unul per text)."""
|
||||
...
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Backend real: fastembed/ONNX #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
class FastEmbedBackend:
|
||||
"""Backend fastembed/ONNX. Lazy-load la constructie.
|
||||
|
||||
Arunca ImportError daca fastembed nu e instalat, sau orice exceptie
|
||||
de la TextEmbedding (download esuat, ONNX incompatibil etc.).
|
||||
Apelantul (_load_engine) prinde aceste exceptii.
|
||||
"""
|
||||
|
||||
def __init__(self, model_name: str = FASTEMBED_MODEL):
|
||||
from fastembed import TextEmbedding # import tardiv -- nu blocheaza la import modul
|
||||
self._model = TextEmbedding(model_name=model_name)
|
||||
|
||||
def embed(self, texts: list[str]) -> list[list[float]]:
|
||||
# fastembed.embed() intoarce un generator de numpy arrays
|
||||
return [vec.tolist() for vec in self._model.embed(texts)]
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Motor de embedding (testabil, backend injectabil) #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
def _cosine_similarity(a: list[float], b: list[float]) -> float:
|
||||
"""Similaritate cosine intre doi vectori. Returneaza 0.0 pe vectori nuli."""
|
||||
dot = sum(x * y for x, y in zip(a, b))
|
||||
na = math.sqrt(sum(x * x for x in a))
|
||||
nb = math.sqrt(sum(x * x for x in b))
|
||||
if na == 0.0 or nb == 0.0:
|
||||
return 0.0
|
||||
return dot / (na * nb)
|
||||
|
||||
|
||||
class EmbeddingEngine:
|
||||
"""Motor de embedding cu corpus indexat si cautare NN cosine.
|
||||
|
||||
Parametri:
|
||||
backend: instanta EmbeddingBackend (real sau mock).
|
||||
None => degradare gratioasa (is_available=False).
|
||||
"""
|
||||
|
||||
def __init__(self, backend: EmbeddingBackend | None = None):
|
||||
self._backend = backend
|
||||
self._corpus_vecs: list[list[float]] = []
|
||||
self._corpus_items: list[dict] = []
|
||||
self._corpus_sig: str | None = None
|
||||
|
||||
def is_available(self) -> bool:
|
||||
"""True daca backend-ul e disponibil si gata de folosire."""
|
||||
return self._backend is not None
|
||||
|
||||
def has_corpus(self) -> bool:
|
||||
"""True daca un corpus a fost indexat (suggest_nearest poate produce ceva)."""
|
||||
return bool(self._corpus_items)
|
||||
|
||||
def corpus_signature(self) -> str | None:
|
||||
"""Semnatura corpusului indexat (None daca gol). Apelantul re-indexeaza
|
||||
doar cand semnatura nomenclatorului s-a schimbat (evita re-embed inutil)."""
|
||||
return self._corpus_sig
|
||||
|
||||
def index_corpus(self, items: list[dict], signature: str | None = None) -> None:
|
||||
"""Vectorizeaza corpus [{denumire, cod}] si il pastreaza in memorie.
|
||||
|
||||
Ignora silentios daca backend-ul lipseste, corpus-ul e gol sau apare
|
||||
orice exceptie la vectorizare (degradare gratioasa).
|
||||
"""
|
||||
self._corpus_vecs = []
|
||||
self._corpus_items = []
|
||||
self._corpus_sig = None
|
||||
|
||||
if not items or not self.is_available():
|
||||
return
|
||||
|
||||
try:
|
||||
texts = [str(item["denumire"]) for item in items]
|
||||
vecs = self._backend.embed(texts)
|
||||
self._corpus_vecs = vecs
|
||||
self._corpus_items = list(items)
|
||||
self._corpus_sig = signature
|
||||
except Exception as exc:
|
||||
log.warning("embeddings: index_corpus esuat: %s", exc)
|
||||
# corpus ramane gol -- suggest_nearest va returna []
|
||||
|
||||
def suggest_nearest(
|
||||
self,
|
||||
denumire: str,
|
||||
top_k: int = 3,
|
||||
) -> list[dict]:
|
||||
"""Returneaza top_k vecini cosine [{cod, is_nul, similaritate}].
|
||||
|
||||
`is_nul` (PRD 5.18 US-005): cand corpusul include exemple NUL (non-operatii),
|
||||
un vecin NUL = semnal de SUPRESIE, nu cod. Default False pe corpusuri vechi
|
||||
fara `is_nul` in itemi. Returneaza [] daca backend-ul lipseste, corpus-ul e gol
|
||||
sau apare orice exceptie (degradare gratioasa -- nu blocheaza ingestia).
|
||||
"""
|
||||
if not self.is_available() or not self._corpus_items:
|
||||
return []
|
||||
|
||||
try:
|
||||
query_vecs = self._backend.embed([str(denumire)])
|
||||
query_vec = query_vecs[0]
|
||||
scored = [
|
||||
{
|
||||
"cod": item["cod"],
|
||||
"is_nul": bool(item.get("is_nul", False)),
|
||||
"similaritate": _cosine_similarity(query_vec, vec),
|
||||
}
|
||||
for item, vec in zip(self._corpus_items, self._corpus_vecs)
|
||||
]
|
||||
scored.sort(key=lambda r: r["similaritate"], reverse=True)
|
||||
return scored[:top_k]
|
||||
except Exception as exc:
|
||||
log.warning("embeddings: suggest_nearest esuat: %s", exc)
|
||||
return []
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Singleton global cu lazy load (API-only, NU worker) #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
_engine: EmbeddingEngine | None = None
|
||||
|
||||
|
||||
def _load_engine() -> EmbeddingEngine:
|
||||
"""Lazy load: construieste engine-ul la prima folosire.
|
||||
|
||||
Captureaza ORICE exceptie la incarcare (import, download, ONNX init)
|
||||
si returneaza un engine degradat (backend=None) -- ingestia continua
|
||||
pe exact+fuzzy, embedding = sugestie dezactivata.
|
||||
"""
|
||||
try:
|
||||
backend = FastEmbedBackend()
|
||||
log.info("embeddings: backend fastembed incarcat (%s)", FASTEMBED_MODEL)
|
||||
return EmbeddingEngine(backend=backend)
|
||||
except ImportError:
|
||||
log.warning(
|
||||
"embeddings: fastembed nu e instalat -- sugestii NN dezactivate"
|
||||
)
|
||||
except Exception as exc:
|
||||
log.warning(
|
||||
"embeddings: incarcare backend esuata (%s) -- sugestii NN dezactivate",
|
||||
exc,
|
||||
)
|
||||
return EmbeddingEngine(backend=None)
|
||||
|
||||
|
||||
def _get_engine() -> EmbeddingEngine:
|
||||
"""Returneaza engine-ul global (lazy-init)."""
|
||||
global _engine
|
||||
if _engine is None:
|
||||
_engine = _load_engine()
|
||||
return _engine
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# API public la nivel de modul (wiring L14-S6) #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
def is_available() -> bool:
|
||||
"""True daca modelul e incarcat si gata de folosire."""
|
||||
return _get_engine().is_available()
|
||||
|
||||
|
||||
def has_corpus() -> bool:
|
||||
"""True daca un corpus a fost indexat in motorul global.
|
||||
|
||||
NU forteaza incarcarea modelului: daca engine-ul nu a fost initializat inca
|
||||
(`_engine is None`), corpus-ul e gol prin definitie -> False, fara cost.
|
||||
Apelantii (ex. enrich_suggestions) folosesc asta ca poarta ieftina inainte de
|
||||
a atinge calea scumpa (is_available/suggest_nearest, care lazy-load ~230MB).
|
||||
"""
|
||||
if _engine is None:
|
||||
return False
|
||||
return _engine.has_corpus()
|
||||
|
||||
|
||||
def corpus_signature() -> str | None:
|
||||
"""Semnatura corpusului global indexat (None daca engine ne-initializat/gol).
|
||||
|
||||
NU forteaza incarcarea modelului: `_engine is None` -> None fara cost.
|
||||
"""
|
||||
if _engine is None:
|
||||
return None
|
||||
return _engine.corpus_signature()
|
||||
|
||||
|
||||
def index_corpus(items: list[dict], signature: str | None = None) -> None:
|
||||
"""Vectorizeaza corpus [{denumire, cod}] in motorul global.
|
||||
|
||||
Silentios pe eroare (degradare gratioasa).
|
||||
"""
|
||||
_get_engine().index_corpus(items, signature=signature)
|
||||
|
||||
|
||||
def suggest_nearest(denumire: str, top_k: int = 3) -> list[dict]:
|
||||
"""Returneaza top_k sugestii [{cod, is_nul, similaritate}] sau [] la eroare.
|
||||
|
||||
Sigur de apelat indiferent de starea backend-ului.
|
||||
"""
|
||||
return _get_engine().suggest_nearest(denumire, top_k=top_k)
|
||||
@@ -1,4 +1,4 @@
|
||||
"""Catalog central de erori AutoPass (PRD 5.4).
|
||||
"""Catalog central de erori AutoPass.
|
||||
|
||||
Singura sursa de adevar care mapeaza fiecare cod de eroare la (problema, fix),
|
||||
cu un helper care construieste obiectul de eroare pe 3 niveluri:
|
||||
@@ -178,6 +178,22 @@ CATALOG: dict[str, dict[str, str]] = {
|
||||
" cererii (request_id) afisat."
|
||||
),
|
||||
},
|
||||
# Coduri de plan (PRD 5.17)
|
||||
"PLAN_LIMITA_LUNARA": {
|
||||
"problema": "Ai atins limita planului Gratuit (60 prestatii/luna)",
|
||||
"fix": (
|
||||
"Treci pe planul Standard sau Pro, sau asteapta inceperea lunii urmatoare."
|
||||
" Numarul de prestatii ramase in luna curenta e in campul cauza."
|
||||
),
|
||||
},
|
||||
"PLAN_FARA_API": {
|
||||
"problema": "Importul prin API e disponibil pe planul Pro",
|
||||
"fix": (
|
||||
"Planul tau curent nu include accesul la API."
|
||||
" Endpoint-ul /v1/prezentari/valideaza ramane disponibil pentru testare fara upgrade."
|
||||
" Contacteaza-ne pentru a face upgrade la planul Pro."
|
||||
),
|
||||
},
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -1,24 +1,18 @@
|
||||
"""Cheie de idempotenta = hash de continut canonic.
|
||||
|
||||
RAR nu are camp nr. comanda si accepta duplicate -> dedup-ul e in sarcina noastra
|
||||
(plan.md sect. 14). Hash stabil peste o reprezentare canonica a prezentarii.
|
||||
RAR nu are camp nr. comanda si accepta duplicate -> dedup-ul e in sarcina noastra.
|
||||
Hash stabil peste o reprezentare canonica a prezentarii.
|
||||
|
||||
Treapta 2 (T9 + OV-2): extrage canonicalize_row + build_key ca helpere publice
|
||||
partajate intre canalul API si canalul import.
|
||||
canonicalize_row + build_key sunt helpere publice partajate intre canalul API si
|
||||
canalul import:
|
||||
- canonicalize_row: normeaza VIN/nr/odometru (strip ".0" Excel coercion) INAINTE
|
||||
de validare (§3.4bis) si INAINTE de cheie.
|
||||
de validare si INAINTE de cheie.
|
||||
- build_key: aplica account_or_default INAINTE de hash (None si 1 => o cheie).
|
||||
Altfel acelasi rand logic din canale diferite (account_id None pe canalul API,
|
||||
1 pe import) ar primi chei diferite -> al doilea FINALIZATA duplicat.
|
||||
|
||||
OV-2 — skew account_id: routerul vechi pasa account_id AS-PASSED (None pe canal API
|
||||
fara auth). Randurile se stocau sub account_or_default=1, dar cheia includea None.
|
||||
Acelasi rand logic din import (account_id=1) dadea cheie diferita -> already_sent
|
||||
rata -> al doilea FINALIZATA. Fix: build_key normalizeaza INTOTDEAUNA la
|
||||
account_or_default inainte de hash.
|
||||
|
||||
Migrare DB productie (OV-2): randurile existente cu cheie-None nu mai sunt gasite de
|
||||
build_key nou. Strategie documentata: dual-lookup la already_sent (incearca cheia
|
||||
noua, apoi cheia legacy). In dev nu exista date reale; la first-deploy productie
|
||||
se poate face recompute-keys o singura data.
|
||||
Randuri vechi cu cheie-None nu sunt gasite de build_key nou: dual-lookup la
|
||||
already_sent (cheia noua, apoi build_key_legacy) sau recompute-keys o singura data.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -46,10 +40,7 @@ def canonicalize_row(raw: dict[str, Any]) -> dict[str, Any]:
|
||||
- data_prestatie: strip (normalizarea la YYYY-MM-DD se face in parser).
|
||||
- prestatii: pastrate ca-atare (rezolvarea e in resolve_prestatii).
|
||||
"""
|
||||
# VIN
|
||||
vin = (raw.get("vin") or "").strip().upper()
|
||||
|
||||
# Nr. inmatriculare
|
||||
nr = (raw.get("nr_inmatriculare") or "").strip().upper()
|
||||
|
||||
# Odometru: strip ".0" Excel float coercion
|
||||
@@ -79,17 +70,23 @@ def canonicalize_row(raw: dict[str, Any]) -> dict[str, Any]:
|
||||
}
|
||||
|
||||
|
||||
def build_key(account_id: int | None, canon: dict[str, Any]) -> str:
|
||||
"""SHA-256 partajat canal-API + canal-import.
|
||||
def build_key(account_id: int | None, canon: dict[str, Any], rar_env: str = "test") -> str:
|
||||
"""SHA-256 partajat canal-API + canal-import, env-aware (PRD 5.20 US-003).
|
||||
|
||||
Aplica account_or_default inainte de hash (OV-2): None si 1 colapseaza la
|
||||
aceeasi cheie => acelasi rand logic din canale diferite nu se trimite de doua ori.
|
||||
Aplica account_or_default inainte de hash: None si 1 colapseaza la aceeasi
|
||||
cheie => acelasi rand logic din canale diferite nu se trimite de doua ori.
|
||||
|
||||
`rar_env` ('test'|'prod') intra in cheie: aceeasi prezentare la test si apoi la
|
||||
prod sunt DOUA trimiteri reale distincte (sisteme RAR separate), nu un duplicat.
|
||||
Default 'test' = back-compat cu apelantii care nu paseaza inca env-ul; toate
|
||||
rutele de ingestie paseaza env-ul rezolvat explicit.
|
||||
"""
|
||||
# Import local ca sa evitam import circular (mapping importa din idempotency via validator)
|
||||
from .mapping import account_or_default
|
||||
acct = account_or_default(account_id)
|
||||
canonic = {
|
||||
"account_id": acct,
|
||||
"rar_env": rar_env,
|
||||
"vin": canon.get("vin", ""),
|
||||
"nr_inmatriculare": canon.get("nr_inmatriculare", ""),
|
||||
"data_prestatie": canon.get("data_prestatie"),
|
||||
@@ -100,25 +97,25 @@ def build_key(account_id: int | None, canon: dict[str, Any]) -> str:
|
||||
return hashlib.sha256(blob.encode("utf-8")).hexdigest()
|
||||
|
||||
|
||||
def idempotency_key(account_id: int | None, prezentare: dict[str, Any]) -> str:
|
||||
"""SHA-256 peste (account_id + campurile semnificative ale prezentarii).
|
||||
def idempotency_key(account_id: int | None, prezentare: dict[str, Any], rar_env: str = "test") -> str:
|
||||
"""SHA-256 peste (account_id + rar_env + campurile semnificative ale prezentarii).
|
||||
|
||||
Wrapper backward-compat peste canonicalize_row + build_key.
|
||||
Exclude obs si b64Image (cosmetice, nu definesc unicitatea declaratiei).
|
||||
|
||||
NOTA: dupa OV-2, account_id=None si account_id=1 produc ACEEASI cheie
|
||||
(via account_or_default in build_key). Randuri vechi cu cheie-None nu sunt
|
||||
NOTA: account_id=None si account_id=1 produc ACEEASI cheie (via
|
||||
account_or_default in build_key). Randuri vechi cu cheie-None nu sunt
|
||||
acoperite automat — dual-lookup sau recompute-keys la migrare productie.
|
||||
"""
|
||||
canon = canonicalize_row(prezentare)
|
||||
return build_key(account_id, canon)
|
||||
return build_key(account_id, canon, rar_env)
|
||||
|
||||
|
||||
def build_key_legacy(account_id: int | None, prezentare: dict[str, Any]) -> str:
|
||||
"""Cheia in formatul vechi (account_id AS-PASSED, fara canonicalize).
|
||||
|
||||
Folosita EXCLUSIV pentru dual-lookup la already_sent pe DB cu randuri vechi
|
||||
(dinainte de T9). Nu folosi pentru randuri noi.
|
||||
Folosita EXCLUSIV pentru dual-lookup la already_sent pe DB cu randuri vechi.
|
||||
Nu folosi pentru randuri noi.
|
||||
"""
|
||||
canonic = {
|
||||
"account_id": account_id,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
"""Parser fisiere xlsx/csv pentru import prezentari (Treapta 2, U1).
|
||||
"""Parser fisiere xlsx/csv pentru import prezentari (Treapta 2).
|
||||
|
||||
Arhitectura 2-treceri (Issue 2, consens cross-model):
|
||||
Arhitectura 2-treceri:
|
||||
Trecerea 1 — read_only=True: dim-check (FileTooLarge) + detectie multi-sheet.
|
||||
Trecerea 2 — normal-mode: header + merged cells + body.
|
||||
Aceasta separare e necesara deoarece openpyxl read_only=True nu vede celule imbinate.
|
||||
@@ -29,7 +29,7 @@ from typing import Any, NamedTuple
|
||||
MAX_ROWS = 5_000
|
||||
MAX_BYTES = 5 * 1024 * 1024 # 5 MB
|
||||
|
||||
# Prag rata None pe o coloana obligatorie -> mesaj formule necalculate (Issue 3)
|
||||
# Prag rata None pe o coloana obligatorie -> mesaj formule necalculate
|
||||
FORMULA_NONE_RATE = 0.6
|
||||
|
||||
# Coloane cheie pentru detectia footer-ului (trim structural)
|
||||
@@ -82,7 +82,7 @@ class ParsedFile(NamedTuple):
|
||||
columns: list[str] # Numele coloanelor detectate (din header)
|
||||
rows: list[dict[str, Any]] # Fiecare rand: {coloana: valoare_bruta}
|
||||
coercion_flags: dict[int, list[str]] # {row_index: [motive needs_review]}
|
||||
formula_columns: list[str] # Coloane cu rata None ridicata (Issue 3)
|
||||
formula_columns: list[str] # Coloane cu rata None ridicata
|
||||
date_col_format: dict[str, str] # {coloana: "DD.MM.YYYY" | "YYYY-MM-DD" | "native" | "ambiguous"}
|
||||
|
||||
|
||||
@@ -230,13 +230,13 @@ def _xlsx_parse_sheet(ws, sheet_name: str) -> ParsedFile:
|
||||
# Trim footer: elimina randuri trailing unde coloanele cheie sunt goale
|
||||
raw_rows = _trim_footer(raw_rows, col_names)
|
||||
|
||||
# Detectie coloane cu formule (rata None, Issue 3)
|
||||
# Detectie coloane cu formule (rata None ridicata)
|
||||
formula_columns = _detect_formula_columns(col_values, len(raw_rows))
|
||||
|
||||
# Detectie format data la nivel de coloana (T10/OV-8)
|
||||
# Detectie format data la nivel de coloana
|
||||
date_col_format = _detect_date_formats(col_values, col_names)
|
||||
|
||||
# Coercion + flags needs_review (T3)
|
||||
# Coercion + flags needs_review
|
||||
coercion_flags: dict[int, list[str]] = {}
|
||||
processed_rows: list[dict[str, Any]] = []
|
||||
for i, row_dict in enumerate(raw_rows):
|
||||
@@ -289,7 +289,7 @@ def _trim_footer(rows: list[dict[str, Any]], col_names: list[str]) -> list[dict[
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Detectie coloane formule (Issue 3) #
|
||||
# Detectie coloane formule #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
def _detect_formula_columns(col_values: dict[str, list[Any]], n_rows: int) -> list[str]:
|
||||
@@ -306,7 +306,7 @@ def _detect_formula_columns(col_values: dict[str, list[Any]], n_rows: int) -> li
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Dezambiguizare data la nivel de coloana (T10 / OV-8) #
|
||||
# Dezambiguizare data la nivel de coloana #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
def _detect_date_formats(col_values: dict[str, list[Any]], col_names: list[str]) -> dict[str, str]:
|
||||
@@ -344,7 +344,7 @@ def _detect_date_formats(col_values: dict[str, list[Any]], col_names: list[str])
|
||||
result[col_name] = "mixed"
|
||||
continue
|
||||
|
||||
# Toate string — detectie format la nivel de coloana (OV-8)
|
||||
# Toate string — detectie format la nivel de coloana
|
||||
fmt = _infer_date_format_from_column(str_vals)
|
||||
result[col_name] = fmt
|
||||
|
||||
@@ -354,7 +354,7 @@ def _detect_date_formats(col_values: dict[str, list[Any]], col_names: list[str])
|
||||
def _infer_date_format_from_column(str_vals: list[str]) -> str:
|
||||
"""Detecteaza formatul datei dintr-o lista de valori string.
|
||||
|
||||
Logica OV-8: daca ORICARE rand are token pozitia-1 > 12 -> coloana e DD-first.
|
||||
Daca ORICARE rand are token pozitia-1 > 12 -> coloana e DD-first.
|
||||
Daca toti zi <= 12 -> ambiguu.
|
||||
"""
|
||||
dd_first_evidence = False
|
||||
@@ -421,7 +421,7 @@ def _split_date(s: str) -> list[str] | None:
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Coercion per rand (T3) #
|
||||
# Coercion per rand #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
def _coerce_row(row_dict: dict[str, Any], col_names: list[str]) -> tuple[dict[str, Any], list[str]]:
|
||||
@@ -682,7 +682,7 @@ def parse_csv(data: bytes) -> ParsedFile:
|
||||
def parse_xlsx(data: bytes, *, sheet_name: str | None = None) -> ParsedFile:
|
||||
"""Parseaza un fisier XLSX.
|
||||
|
||||
Arhitectura 2-treceri (Issue 2):
|
||||
Arhitectura 2-treceri:
|
||||
1. read_only=True: dim-check + detectie multi-sheet
|
||||
2. normal-mode: header + merged cells + body
|
||||
|
||||
|
||||
19
app/main.py
19
app/main.py
@@ -1,9 +1,7 @@
|
||||
"""Aplicatia FastAPI: API v1 + dashboard web + /healthz + /metrics.
|
||||
|
||||
Worker-ul ruleaza ca PROCES SEPARAT (python -m app.worker), NU ca task aici
|
||||
(plan.md sect. 4: un worker mort nu trebuie sa lase containerul "sanatos").
|
||||
|
||||
Pornire dev: uvicorn app.main:app --reload
|
||||
Worker-ul ruleaza ca PROCES SEPARAT (python -m app.worker), NU ca task aici:
|
||||
un worker mort nu trebuie sa lase containerul "sanatos".
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -44,7 +42,7 @@ from .web.session import AdminRequired, LoginRequired
|
||||
async def lifespan(app: FastAPI):
|
||||
install_log_redaction()
|
||||
# Fail-fast: o cheie Fernet setata dar invalida opreste pornirea cu mesaj clar,
|
||||
# in loc de 500 brut la primul POST /v1/prezentari (cazul reprodus din VFP).
|
||||
# in loc de 500 brut la primul POST /v1/prezentari.
|
||||
validate_creds_key()
|
||||
init_db()
|
||||
yield
|
||||
@@ -61,7 +59,7 @@ app.add_middleware(
|
||||
https_only=settings.session_https_only,
|
||||
same_site="strict",
|
||||
)
|
||||
# US-002: request_id pe fiecare cerere. Adaugat dupa SessionMiddleware -> ruleaza
|
||||
# request_id pe fiecare cerere. Adaugat dupa SessionMiddleware -> ruleaza
|
||||
# OUTERMOST (add_middleware prepend), deci `X-Request-ID` se pune pe TOATE raspunsurile,
|
||||
# inclusiv 401/404/422/500 produse mai in interior.
|
||||
app.add_middleware(RequestIDMiddleware)
|
||||
@@ -97,13 +95,11 @@ async def validation_exception_handler(request: Request, exc: RequestValidationE
|
||||
|
||||
@app.exception_handler(Exception)
|
||||
async def unhandled_exception_handler(request: Request, exc: Exception) -> JSONResponse:
|
||||
"""Orice excepție neprinsa -> 500 STRUCTURAT (3 niveluri, PRD 5.4) in loc de 500 brut.
|
||||
"""Orice excepție neprinsa -> 500 STRUCTURAT din catalog in loc de 500 brut.
|
||||
|
||||
Body = envelope-ul standard din catalog (6 chei: field/cod/problema/cauza/fix/message)
|
||||
+ `request_id` — fara traceback, fara mesaj de excepție brut, fara creds. Traceback-ul
|
||||
complet + ruta + cont + request_id se scriu DOAR in jurnal (redactate prin scrub_text).
|
||||
Handlerele specifice (LoginRequired/AdminRequired/CSRF/RequestValidationError/HTTPException)
|
||||
raman neatinse — acesta prinde doar ce nu are handler dedicat.
|
||||
"""
|
||||
request_id = getattr(request.state, "request_id", None) or request_id_var.get()
|
||||
try:
|
||||
@@ -144,9 +140,8 @@ app.include_router(admin_router)
|
||||
def healthz() -> dict:
|
||||
"""Sanatate: worker viu + ultimul login RAR reusit + adancime coada.
|
||||
|
||||
Pica (200 cu ok=False / sau folosit de orchestrator) cand worker-ul e mort
|
||||
-> semnal de restart (plan.md sect. 8). Intoarce 200 mereu cu detalii;
|
||||
orchestratorul decide pe campul `worker_alive`.
|
||||
Intoarce 200 mereu cu detalii; orchestratorul decide restartul pe campul
|
||||
`worker_alive`.
|
||||
"""
|
||||
settings = get_settings()
|
||||
conn = get_connection()
|
||||
|
||||
343
app/mapping.py
343
app/mapping.py
@@ -1,7 +1,7 @@
|
||||
"""Mapare operatie ROAAUTO -> cod prestatie RAR + fuzzy lookup pentru editor.
|
||||
|
||||
Contract (varianta hibrida, decis 2026-06-15): un item de prestatie poate veni
|
||||
fie cu `cod_prestatie` (cod RAR direct, ca pana acum), fie cu `cod_op_service`
|
||||
Contract (varianta hibrida): un item de prestatie poate veni
|
||||
fie cu `cod_prestatie` (cod RAR direct), fie cu `cod_op_service`
|
||||
(cod intern ROAAUTO) + `denumire`. La ingestie incercam sa rezolvam codul intern
|
||||
prin `operations_mapping`; daca nu exista mapare -> submission `needs_mapping`
|
||||
(nu se trimite la RAR), iar operatia apare in editorul web unde userul o mapeaza
|
||||
@@ -14,7 +14,9 @@ unit-testabile direct. Cele cu `conn` sunt helpere de persistenta.
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import hashlib
|
||||
import json
|
||||
import re
|
||||
import unicodedata
|
||||
from typing import Any
|
||||
|
||||
@@ -48,6 +50,60 @@ def normalize_for_match(value: object) -> str:
|
||||
return " ".join(s.upper().split())
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Pre-filtru determinist non-operatii (NUL) — US-001 PRD 5.18 #
|
||||
# --------------------------------------------------------------------------- #
|
||||
#
|
||||
# Masuratoarea k-NN (memorie test-precizie-knn-embeddings) arata recall NUL doar
|
||||
# 64%: gunoiul evident (ITP, plata, discount, nr. inmatriculare, tractare) scapa
|
||||
# semantic ca OE-1. Un pre-filtru text/regex il marcheaza NUL INAINTE de k-NN.
|
||||
#
|
||||
# Garantie: ZERO fals-pozitiv pe operatii reale. Regulile au fost calibrate pe
|
||||
# `docs/operatii-service/*.csv` (toate aparitiile distincte). Triggerele NEambigue
|
||||
# (ITP, ACHITAT/PLATA, DISCOUNT/REDUCERE, TAXA) sunt neconditionate (0 FP masurat).
|
||||
# Triggerele AMBIGUE (TRACTARE, NR INMATRICULARE + pattern placuta) apar si in
|
||||
# operatii reale ("D/R CARLIG TRACTARE", "D/R ELECTROMOTOR CT 44 MKY") -> sunt
|
||||
# ECRANATE de un context de piesa/operatie (`_NUL_CTX_PIESA`).
|
||||
|
||||
# Trigger-uri neambigue (substring/regex pe text normalizat).
|
||||
_NUL_ITP = re.compile(r"(?:\bITP\b|\d\s*X\s*ITP|X\s*ITP\b|\bITP[.,])")
|
||||
_NUL_PLATA = re.compile(r"\b(ACHITAT|ACHITARE|PLATA|PLATIT|PLATIRE)\b")
|
||||
_NUL_DISCOUNT = re.compile(r"\b(DISCOUNT|REDUCERE)\b")
|
||||
_NUL_TAXA = re.compile(r"\bTAXA\b")
|
||||
|
||||
# Trigger-uri ambigue — valide ca NUL DOAR in absenta unui context de piesa.
|
||||
_NUL_TRACTARE = re.compile(r"\b(TRACTARE|TRACTARI)\b")
|
||||
_NUL_NR_PLACUTA = re.compile(
|
||||
r"(\bNR\s+INMATRICULARE\b|\bNUMAR\s+INMATRICULARE\b|\b[A-Z]{1,2}\s?\d{2,3}\s?[A-Z]{3}\b)"
|
||||
)
|
||||
# Daca apare oricare cuvant de aici, TRACTARE/placuta e nume de piesa sau operatie
|
||||
# reala (carlig/capac de tractare, suport placuta, placuta lipita la o reparatie).
|
||||
_NUL_CTX_PIESA = re.compile(
|
||||
r"\b(D/R|D-R|CARLIG|CAPAC|BARA|PROTECTIE|MONTAT|MONTAJ|DEMONTAT|INLOCUIT|"
|
||||
r"INLOCUIRE|REPARAT|REPARATIE|VOPSIT|SCHIMBAT|SUPORT)\b"
|
||||
)
|
||||
|
||||
|
||||
def prefiltru_nul(denumire: object) -> bool:
|
||||
"""True daca operatia e gunoi evident (non-operatie de service) -> NUL determinist.
|
||||
|
||||
Ruleaza INAINTE de k-NN/embeddings in `enrich_suggestions` (US-006). Pur, fara DB.
|
||||
Zero fals-pozitiv pe operatii reale (vezi comentariul de mai sus + tests).
|
||||
"""
|
||||
text = normalize_for_match(denumire)
|
||||
if not text:
|
||||
return False
|
||||
# Neambigue: 0 FP masurat, fara ecranare.
|
||||
if _NUL_ITP.search(text) or _NUL_PLATA.search(text) or _NUL_DISCOUNT.search(text) or _NUL_TAXA.search(text):
|
||||
return True
|
||||
# Ambigue: doar daca NU e context de piesa.
|
||||
if _NUL_CTX_PIESA.search(text):
|
||||
return False
|
||||
if _NUL_TRACTARE.search(text) or _NUL_NR_PLACUTA.search(text):
|
||||
return True
|
||||
return False
|
||||
|
||||
|
||||
def suggest_codes(
|
||||
denumire: object,
|
||||
nomenclator: list[dict],
|
||||
@@ -87,7 +143,7 @@ def suggest_codes(
|
||||
]
|
||||
|
||||
|
||||
# Prefixul pus pe `cod_sursa` cand un item e rezolvat printr-o regula text (US-010).
|
||||
# Prefixul pus pe `cod_sursa` cand un item e rezolvat printr-o regula text.
|
||||
# Forma: "text_rule:<pattern original al regulii castigatoare>". Payload-harmless —
|
||||
# RAR citeste doar `cod_prestatie`; `cod_sursa` ramane in payload_json fara efect.
|
||||
COD_SURSA_TEXT_RULE_PREFIX = "text_rule:"
|
||||
@@ -111,10 +167,10 @@ def _rezolva_din_reguli_text(
|
||||
`valid_codes` (cand `valid_codes` e setat), nu intoarcem un cod invalid ->
|
||||
(None, None, None) (operatia ramane nemapata), coerent cu garda din `resolve_prestatii`.
|
||||
|
||||
Pattern-ul intors e cel ORIGINAL al regulii (pentru telemetrie US-010), nu cel
|
||||
Pattern-ul intors e cel ORIGINAL al regulii (pentru telemetrie), nu cel
|
||||
normalizat folosit la match. `auto_send` = flagul regulii castigatoare: cand e
|
||||
falsy (DEFAULT 0, decizia CEO de siguranta) randul trebuie TINUT pentru verificare
|
||||
umana, nu trimis automat la RAR (blast radius substring + FINALIZATA ireversibil).
|
||||
falsy (DEFAULT 0, de siguranta) randul trebuie TINUT pentru verificare umana, nu
|
||||
trimis automat la RAR (blast radius substring + FINALIZATA ireversibil).
|
||||
"""
|
||||
if not text_rules:
|
||||
return None, None, None
|
||||
@@ -136,7 +192,7 @@ def _rezolva_din_reguli_text(
|
||||
|
||||
|
||||
def text_rule_hits(resolved: list[dict] | None) -> list[dict]:
|
||||
"""Extrage din itemii rezolvati cei care au primit cod dintr-o regula text (US-010).
|
||||
"""Extrage din itemii rezolvati cei care au primit cod dintr-o regula text.
|
||||
|
||||
Intoarce [{pattern, cod_prestatie}] pentru fiecare item al carui `cod_sursa`
|
||||
incepe cu `COD_SURSA_TEXT_RULE_PREFIX`. Pur (fara DB); apelantii cu `conn` il
|
||||
@@ -154,7 +210,7 @@ def text_rule_hits(resolved: list[dict] | None) -> list[dict]:
|
||||
|
||||
|
||||
def text_rules_overlap(pattern: str, existing_rules: list[dict] | None) -> list[dict]:
|
||||
"""Reguli text existente care se SUPRAPUN cu `pattern` (US-011, avertisment neblocant).
|
||||
"""Reguli text existente care se SUPRAPUN cu `pattern` (avertisment neblocant).
|
||||
|
||||
Overlap = pattern-ul nou normalizat (`normalize_for_match`) e substring al unei
|
||||
reguli existente SAU invers (oricare directie). Pur, determinist, fara DB.
|
||||
@@ -192,9 +248,9 @@ def resolve_prestatii(
|
||||
- item fara cod, fara mapare si fara regula text -> ramane nemapat.
|
||||
- item cu `cod_prestatie` NECUNOSCUT in nomenclator -> tratat ca operatie de
|
||||
mapat: il promovam la `cod_op_service` (daca nu exista deja) ca sa intre in
|
||||
fluxul needs_mapping. Confirmat live (2026-06-23): RAR accepta NUMAI coduri
|
||||
din nomenclator (coloana COD_PRESTATIE max 5 car.); un cod necunoscut da
|
||||
HTTP 500 si RECORD PARTIAL la RAR (terminal) -> nu-l trimitem niciodata raw.
|
||||
fluxul needs_mapping. RAR accepta NUMAI coduri din nomenclator (coloana
|
||||
COD_PRESTATIE max 5 car.); un cod necunoscut da HTTP 500 si RECORD PARTIAL
|
||||
la RAR (terminal) -> nu-l trimitem niciodata raw.
|
||||
|
||||
Precedenta (stricta): `cod_prestatie` direct valid > mapare exacta `cod_op_service`
|
||||
in `mapping` > reguli text > nemapat. Regulile text se incearca DOAR cand nu exista
|
||||
@@ -217,8 +273,8 @@ def resolve_prestatii(
|
||||
unmapped: list[dict] = []
|
||||
for item in prestatii or []:
|
||||
it = dict(item)
|
||||
# Curata adnotarile aditive ale rezolvarii (cod_sursa US-010 + flagul de
|
||||
# hold pe regula auto_send=0): se recalculeaza de la zero la fiecare rezolvare.
|
||||
# Curata adnotarile aditive ale rezolvarii (cod_sursa + flagul de hold pe
|
||||
# regula auto_send=0): se recalculeaza de la zero la fiecare rezolvare.
|
||||
# Altfel, un item re-rezolvat acum prin alta cale (ex. mapare exacta) ar pastra
|
||||
# un cod_sursa/flag stale din payload -> telemetrie falsa + hold gresit.
|
||||
it.pop("cod_sursa", None)
|
||||
@@ -246,13 +302,11 @@ def resolve_prestatii(
|
||||
)
|
||||
if cod_regula is not None:
|
||||
it["cod_prestatie"] = cod_regula
|
||||
# Adnotare aditiva (US-010): marcheaza ca rezolvat-prin-regula cu
|
||||
# pattern-ul sursa. Payload-harmless (RAR citeste doar cod_prestatie).
|
||||
# Adnotare aditiva: marcheaza ca rezolvat-prin-regula cu pattern-ul
|
||||
# sursa. Payload-harmless (RAR citeste doar cod_prestatie).
|
||||
it["cod_sursa"] = f"{COD_SURSA_TEXT_RULE_PREFIX}{pattern_regula or ''}"
|
||||
# Siguranta CEO (US-001): regula cu auto_send=0 rezolva codul dar
|
||||
# TINE randul pentru verificare umana (has_no_auto_send -> True).
|
||||
if not auto_send_regula:
|
||||
it["regula_fara_autosend"] = True
|
||||
# US-001 (PRD 5.11): regula_fara_autosend nu se mai seteaza;
|
||||
# auto_send nu mai tine randul (has_no_auto_send neutralizat).
|
||||
else:
|
||||
it["cod_prestatie"] = None
|
||||
unmapped.append({"cod_op_service": op, "denumire": it.get("denumire")})
|
||||
@@ -273,7 +327,7 @@ def account_or_default(account_id: int | None) -> int:
|
||||
def account_scope_clause(account_id: int) -> tuple[str, list]:
|
||||
"""Fragment SQL + params pentru filtrarea pe cont in tabele cu account_id nullable.
|
||||
|
||||
Aplica regula: NULL apartine contului 1 (legacy/OV-2).
|
||||
Aplica regula: NULL apartine contului 1 (legacy).
|
||||
Foloseste DOAR pe submissions (account_id NULLABLE).
|
||||
NU folosi pe operations_mapping (account_id NOT NULL) — acolo WHERE account_id=? simplu.
|
||||
"""
|
||||
@@ -356,7 +410,7 @@ def load_mapping(conn, account_id: int | None) -> dict[str, str]:
|
||||
def load_mapping_meta(conn, account_id: int | None) -> dict[str, dict]:
|
||||
"""{cod_op_service -> {cod_prestatie, auto_send}} pentru un cont.
|
||||
|
||||
T6/OV-1: varianta extinsa care include si flagul auto_send per operatie.
|
||||
Varianta extinsa care include si flagul auto_send per operatie.
|
||||
"""
|
||||
acct = account_or_default(account_id)
|
||||
rows = conn.execute(
|
||||
@@ -379,7 +433,7 @@ def classify_prezentare(
|
||||
"""Helper pur de clasificare: reproduce EXACT logica create_prezentari fara DB/efecte.
|
||||
|
||||
Apelat de AMBELE rute (POST /v1/prezentari si POST /v1/prezentari/valideaza) pentru
|
||||
a garanta acelasi verdict — invariantul de corectitudine dry-run (PRD 5.2).
|
||||
a garanta acelasi verdict — invariantul de corectitudine dry-run.
|
||||
|
||||
Intoarce {"status", "rar_error", "resolved", "unmapped", "errors", "content"}.
|
||||
"content" = copia actualizata (VIN/nr canonicalizat + prestatii rezolvate).
|
||||
@@ -410,14 +464,9 @@ def classify_prezentare(
|
||||
if errors:
|
||||
status = "needs_data"
|
||||
rar_error = json.dumps(errors, ensure_ascii=False)
|
||||
elif has_no_auto_send(resolved, mapping_meta):
|
||||
status = "needs_mapping"
|
||||
mesaj = "cod mapat cu auto_send=0; review manual inainte de trimitere"
|
||||
rar_error = json.dumps(
|
||||
{"auto_send": mesaj, **err_mod.eroare("AUTO_SEND_OPRIT", cauza=mesaj)},
|
||||
ensure_ascii=False,
|
||||
)
|
||||
else:
|
||||
# US-001 (PRD 5.11): ramura AUTO_SEND_OPRIT eliminata.
|
||||
# Un cod rezolvat (mapare exacta sau regula text) -> queued direct.
|
||||
status = "queued"
|
||||
rar_error = None
|
||||
|
||||
@@ -432,20 +481,14 @@ def classify_prezentare(
|
||||
|
||||
|
||||
def has_no_auto_send(resolved: list[dict], mapping_meta: dict[str, dict]) -> bool:
|
||||
"""Verifica daca vreun item rezolvat are auto_send=0 (mapare exacta SAU regula text).
|
||||
"""Neutralizat dupa US-001 (PRD 5.11): auto_send nu mai tine randuri in needs_mapping.
|
||||
|
||||
T6/OV-1: un cod nou-mapat (operations_mapping) cu auto_send=0 nu trebuie trimis automat.
|
||||
PRD 5.8 US-001 (decizia CEO): la fel pentru un item rezolvat printr-o REGULA TEXT cu
|
||||
auto_send=0 — marcat de `resolve_prestatii` cu `regula_fara_autosend`. In ambele cazuri
|
||||
randul ramane needs_mapping (review manual) pana cand operatorul activeaza „In coada".
|
||||
Items cu cod_prestatie direct (nu via cod_op_service/regula) nu sunt afectate.
|
||||
Simbolul este PASTRAT (importat in routes.py si import_router.py); stergerea
|
||||
ar produce ImportError la boot. Functia intoarce mereu False — codul rezolvat
|
||||
intra direct in queued, indiferent de valoarea auto_send din mapping_meta.
|
||||
|
||||
Coloanele DB raman cu default=1 (migrare non-distructiva).
|
||||
"""
|
||||
for item in resolved:
|
||||
if item.get("regula_fara_autosend"):
|
||||
return True
|
||||
op = (item.get("cod_op_service") or "").strip()
|
||||
if op and op in mapping_meta and not mapping_meta[op]["auto_send"]:
|
||||
return True
|
||||
return False
|
||||
|
||||
|
||||
@@ -457,7 +500,7 @@ def pending_unmapped(conn, account_id=None) -> list[dict]:
|
||||
footgun (scurge cross-account) si e rezervat exclusiv pentru dashboard-ul intern.
|
||||
|
||||
account_id=int: filtreaza in SQL pe cont inclusiv randuri legacy (account_id IS NULL
|
||||
apartine contului 1, OV-2). Filtrarea in SQL, nu post-hoc in Python.
|
||||
apartine contului 1). Filtrarea in SQL, nu post-hoc in Python.
|
||||
"""
|
||||
nomenclator = load_nomenclator(conn)
|
||||
if account_id is not None:
|
||||
@@ -496,10 +539,18 @@ def pending_unmapped(conn, account_id=None) -> list[dict]:
|
||||
entry["denumire"] = item.get("denumire")
|
||||
entry["_ids"].add(r["id"])
|
||||
|
||||
# Indexeaza corpusul embeddings o data inainte de bucla (no-op cand flagul e off).
|
||||
ensure_embeddings_corpus(conn, nomenclator)
|
||||
|
||||
out: list[dict] = []
|
||||
for entry in agg.values():
|
||||
entry["blocked"] = len(entry.pop("_ids"))
|
||||
entry["suggestions"] = suggest_codes(entry["denumire"], nomenclator, limit=5)
|
||||
# L14-S6: imbogatire sugestii cu GOLD partajat > SILVER > embeddings (Eng-F2).
|
||||
# SUGGESTION-ONLY: nu intra in resolve_prestatii/load_mapping (#13).
|
||||
enriched = enrich_suggestions(conn, entry["denumire"])
|
||||
entry["sugestie_principala"] = enriched["sugestie_principala"]
|
||||
entry["surse_sugestie"] = enriched["surse"]
|
||||
out.append(entry)
|
||||
out.sort(key=lambda e: (-e["blocked"], e["cod_op_service"]))
|
||||
return out
|
||||
@@ -574,12 +625,180 @@ def delete_text_rule(conn, account_id: int | None, pattern: str) -> None:
|
||||
)
|
||||
|
||||
|
||||
# Prag minim de similaritate cosine pentru sugestia din embeddings NN.
|
||||
# Sub acest scor, sugestia NN e prea incerta si nu o afisam (previne recomandari
|
||||
# irelevante cand corpus-ul e mic sau neindexat corect).
|
||||
EMB_MIN_SIMILARITATE = 0.5
|
||||
|
||||
|
||||
def _corpus_signature_silver(rows: list) -> str:
|
||||
"""Semnatura stabila a corpusului SILVER (mapping_suggestions) pentru cache.
|
||||
|
||||
Hash pe (denumire_normalizata, cod, is_nul) sortat -> se schimba la orice
|
||||
add/remove/redenumire/relabel, ramane stabila altfel (evita re-embed inutil).
|
||||
"""
|
||||
triples = sorted(
|
||||
(str(r["denumire_normalizata"] or ""), str(r["cod_prestatie"] or ""), int(r["is_nul"] or 0))
|
||||
for r in rows
|
||||
)
|
||||
blob = "".join(f"{d}|{c}|{n}" for d, c, n in triples)
|
||||
return hashlib.sha256(blob.encode("utf-8")).hexdigest()
|
||||
|
||||
|
||||
def ensure_embeddings_corpus(conn, nomenclator: list[dict] | None = None) -> None:
|
||||
"""Construieste/actualizeaza corpusul embeddings din corpusul ETICHETAT (PRD 5.18 US-005).
|
||||
|
||||
Sursa corpusului = `mapping_suggestions` (SILVER): exemple reale etichetate
|
||||
{denumire_normalizata -> cod, is_nul}, NU cele 18 categorii generice din
|
||||
`nomenclator_rar`. k-NN peste exemple reale e net mai precis (94.3% acord LLM).
|
||||
Parametrul `nomenclator` e pastrat pentru compatibilitatea apelantilor, dar nu mai
|
||||
e folosit ca sursa.
|
||||
|
||||
Gated pe `AUTOPASS_EMBEDDINGS_ENABLED` (default ON; OFF in teste): cand e
|
||||
dezactivat, e un no-op total -> /mapari instant + suita de teste rapida.
|
||||
|
||||
Cand e activat: indexeaza corpusul o singura data (lazy-load modelul ~230MB la
|
||||
prima chemare), re-indexeaza doar cand semnatura corpusului SILVER s-a schimbat.
|
||||
Itemii NUL (is_nul=1, cod NULL) raman in corpus: un vecin NUL e semnal de supresie
|
||||
(US-006). Degradare gratioasa: orice eroare lasa corpusul gol -> enrich cade pe restul.
|
||||
"""
|
||||
from .config import get_settings
|
||||
if not get_settings().embeddings_enabled:
|
||||
return
|
||||
try:
|
||||
from . import embeddings as _emb
|
||||
rows = conn.execute(
|
||||
"SELECT denumire_normalizata, cod_prestatie, is_nul FROM mapping_suggestions"
|
||||
).fetchall()
|
||||
if not rows:
|
||||
return
|
||||
sig = _corpus_signature_silver(rows)
|
||||
if _emb.corpus_signature() == sig and _emb.has_corpus():
|
||||
return # deja indexat pe acelasi corpus SILVER -> nimic de facut
|
||||
items = [
|
||||
{
|
||||
"denumire": str(r["denumire_normalizata"]),
|
||||
"cod": (str(r["cod_prestatie"]) if r["cod_prestatie"] is not None else None),
|
||||
"is_nul": bool(r["is_nul"]),
|
||||
}
|
||||
for r in rows
|
||||
if r["denumire_normalizata"]
|
||||
]
|
||||
_emb.index_corpus(items, signature=sig)
|
||||
except Exception:
|
||||
pass # degradare gratioasa (#16b): esecul indexarii nu blocheaza editorul
|
||||
|
||||
|
||||
def enrich_suggestions(
|
||||
conn,
|
||||
denumire: str | None,
|
||||
*,
|
||||
include_embeddings: bool = True,
|
||||
) -> dict:
|
||||
"""Imbogateste sugestiile cu GOLD partajat, SILVER LLM si embeddings NN.
|
||||
|
||||
Precedenta Eng-F2 (pentru sugestie-only, nu auto-send):
|
||||
shared GOLD > SILVER > embeddings
|
||||
|
||||
(Account GOLD = operations_mapping propriu = deja rezolvat inainte de needs_mapping;
|
||||
nu apare in needs_mapping, deci nu e in precedenta de sugestie.)
|
||||
|
||||
Ordine completa (PRD 5.18 US-006):
|
||||
pre-filtru NUL determinist -> (daca NUL: fara cod, `surse['nul']=True`)
|
||||
altfel GOLD partajat > exact (SILVER) > k-NN embeddings.
|
||||
|
||||
Returneaza:
|
||||
{
|
||||
'sugestie_principala': {'cod_prestatie': str, 'sursa': str} | None,
|
||||
'surse': {'gold_partajat': str|None, 'silver': str|None, 'embedding': str|None, 'nul': bool}
|
||||
}
|
||||
|
||||
INVARIANTE:
|
||||
- Toate sursele = SUGGESTION-ONLY. NU intra in resolve_prestatii/load_mapping (#13).
|
||||
- Pre-filtru NUL (US-001) ruleaza PRIMUL: gunoiul evident (ITP/plata/discount...) e
|
||||
marcat non-operatie INAINTE de k-NN, fara sugestie de cod.
|
||||
- SILVER cu is_nul=1 (non-operatie/gunoi) NU produce sugestie (#4); vecin k-NN NUL idem.
|
||||
- Degradare gratioasa pe embeddings (#16b): daca motorul nu e disponibil sau arunca,
|
||||
returneaza sugestia disponibila din celelalte surse, fara exceptie.
|
||||
- Import local shared_store/embeddings: evita ciclu la import-time (shared_store
|
||||
importa normalize_for_match din mapping).
|
||||
"""
|
||||
sugestie_principala: dict | None = None
|
||||
surse: dict = {"gold_partajat": None, "silver": None, "embedding": None, "nul": False}
|
||||
|
||||
if not denumire:
|
||||
return {"sugestie_principala": sugestie_principala, "surse": surse}
|
||||
|
||||
# 0. Pre-filtru NUL determinist (US-001) INAINTE de orice k-NN/lookup: non-operatie
|
||||
# evidenta -> fara cod, scurtcircuit (nu interogheaza embeddings/SILVER pe gunoi).
|
||||
if prefiltru_nul(denumire):
|
||||
surse["nul"] = True
|
||||
return {"sugestie_principala": None, "surse": surse}
|
||||
|
||||
# Colecteaza TOATE sursele (fara short-circuit) in `surse`: editorul le poate afisa
|
||||
# toate, independent de care castiga ca sugestie principala.
|
||||
# Precedenta Eng-F2 se aplica DOAR la alegerea sugestiei_principale.
|
||||
|
||||
# 1. GOLD partajat cross-account (validat de om, cel mai de incredere)
|
||||
try:
|
||||
from .shared_store import lookup_shared_gold
|
||||
row_gold = lookup_shared_gold(conn, denumire)
|
||||
if row_gold and row_gold["cod_prestatie"]:
|
||||
surse["gold_partajat"] = str(row_gold["cod_prestatie"])
|
||||
except Exception:
|
||||
pass # degradare gratioasa
|
||||
|
||||
# 2. SILVER LLM (bootstrap, nevalidat de om; is_nul = supresie)
|
||||
try:
|
||||
from .shared_store import lookup_suggestion
|
||||
row_silver = lookup_suggestion(conn, denumire)
|
||||
if row_silver and not row_silver["is_nul"] and row_silver["cod_prestatie"]:
|
||||
surse["silver"] = str(row_silver["cod_prestatie"])
|
||||
except Exception:
|
||||
pass # degradare gratioasa
|
||||
|
||||
# 3. Embeddings NN (similaritate semantica, degradare gratioasa #16b)
|
||||
if include_embeddings:
|
||||
try:
|
||||
from . import embeddings as _emb
|
||||
# Poarta IEFTINA: nu atinge is_available()/suggest_nearest cand corpus-ul
|
||||
# e gol — `is_available()` lazy-load-eaza modelul de ~230MB (30-120s in
|
||||
# thread-ul de cerere). Corpusul se construieste de apelant prin
|
||||
# ensure_embeddings_corpus (gated pe AUTOPASS_EMBEDDINGS_ENABLED); cand
|
||||
# flagul e off, has_corpus() ramane False si calea e un no-op real.
|
||||
if _emb.has_corpus():
|
||||
# F1 (US-005): corpusul k-NN e text NORMALIZAT (denumire_normalizata),
|
||||
# deci query-ul TREBUIE normalizat la fel — altfel cosine degradeaza si
|
||||
# nu mai e configul sub care s-a masurat 94.3%.
|
||||
nn = _emb.suggest_nearest(normalize_for_match(denumire), top_k=1)
|
||||
# Prag minim: similaritate prea mica = sugestie inutila.
|
||||
# Evita recomandari irelevante cand corpus-ul e mic/partial.
|
||||
if nn and nn[0].get("similaritate", 0) >= EMB_MIN_SIMILARITATE:
|
||||
if nn[0].get("is_nul"):
|
||||
# Vecin NUL (non-operatie) = semnal de SUPRESIE, nu cod (US-006).
|
||||
surse["nul"] = True
|
||||
elif nn[0].get("cod"):
|
||||
surse["embedding"] = str(nn[0]["cod"])
|
||||
except Exception:
|
||||
pass # degradare gratioasa (#16b): motorul absent nu blocheaza
|
||||
|
||||
# Alege sugestia principala in ordinea de precedenta: GOLD > SILVER > embeddings
|
||||
if surse["gold_partajat"]:
|
||||
sugestie_principala = {"cod_prestatie": surse["gold_partajat"], "sursa": "gold_partajat"}
|
||||
elif surse["silver"]:
|
||||
sugestie_principala = {"cod_prestatie": surse["silver"], "sursa": "silver"}
|
||||
elif surse["embedding"]:
|
||||
sugestie_principala = {"cod_prestatie": surse["embedding"], "sursa": "embedding"}
|
||||
|
||||
return {"sugestie_principala": sugestie_principala, "surse": surse}
|
||||
|
||||
|
||||
def _emite_text_rule_hits(conn, account_id: int, submission_id: int, resolved: list[dict] | None) -> None:
|
||||
"""Emite `text_rule_hit` in app_events pentru fiecare item rezolvat prin regula text.
|
||||
|
||||
US-010: telemetrie „ce regula a rezolvat ce submission". Best-effort (log_event
|
||||
inghite exceptiile). Context = {submission_id, account_id, pattern, cod_prestatie} —
|
||||
fara PII (pattern + cod nu sunt PII). Import local: evita orice risc de ciclu la import.
|
||||
Telemetrie „ce regula a rezolvat ce submission". Best-effort (log_event inghite
|
||||
exceptiile). Context = {submission_id, account_id, pattern, cod_prestatie} — fara
|
||||
PII (pattern + cod nu sunt PII). Import local: evita orice risc de ciclu la import.
|
||||
"""
|
||||
hits = text_rule_hits(resolved)
|
||||
if not hits:
|
||||
@@ -604,25 +823,25 @@ def reresolve_account(conn, account_id: int | None, batch_id: int | None = None)
|
||||
"""Re-rezolva submission-urile `needs_mapping` ale unui cont dupa o noua mapare.
|
||||
|
||||
Pentru fiecare: aplica maparea curenta; daca nu mai raman op-uri nemapate ->
|
||||
ruleaza validarea de continut (T3) si trece pe `queued` (sau `needs_data` cu
|
||||
ruleaza validarea de continut si trece pe `queued` (sau `needs_data` cu
|
||||
motiv), resetand backoff-ul. Daca raman nemapate, ramane `needs_mapping` cu
|
||||
motivul actualizat. Intoarce {requeued, still_blocked, needs_data, review_manual}.
|
||||
|
||||
T6/OV-1: auto_send=0 pe un cod nou-mapat -> nu trece pe 'queued' (ramane
|
||||
'needs_mapping' cu motiv "review manual"); previne FINALIZATA eronat permanent.
|
||||
auto_send=0 pe un cod nou-mapat -> nu trece pe 'queued' (ramane 'needs_mapping'
|
||||
cu motiv "review manual"); previne FINALIZATA eronat permanent.
|
||||
|
||||
T7: batch_id != None -> scope la seria comitata (NU cross-batch).
|
||||
batch_id is None -> re-rezolva toti (canal API, batch_id IS NULL inclus).
|
||||
batch_id != None -> scope la seria comitata (NU cross-batch).
|
||||
batch_id is None -> re-rezolva toti (canal API, batch_id IS NULL inclus).
|
||||
"""
|
||||
acct = account_or_default(account_id)
|
||||
mapping_meta = load_mapping_meta(conn, acct)
|
||||
mapping = {op: meta["cod_prestatie"] for op, meta in mapping_meta.items()}
|
||||
valid_codes = load_nomenclator_codes(conn) or None
|
||||
# T2: incarca regulile text O DATA, inainte de bucla pe randuri.
|
||||
# Incarca regulile text O DATA, inainte de bucla pe randuri.
|
||||
text_rules = load_text_rules(conn, acct)
|
||||
|
||||
if batch_id is not None:
|
||||
# T7: scope la batch-ul specificat (import commit explicit).
|
||||
# Scope la batch-ul specificat (import commit explicit).
|
||||
# NU atinge randuri din alte batches sau din feed API.
|
||||
rows = conn.execute(
|
||||
"SELECT id, payload_json FROM submissions "
|
||||
@@ -631,8 +850,8 @@ def reresolve_account(conn, account_id: int | None, batch_id: int | None = None)
|
||||
).fetchall()
|
||||
else:
|
||||
# POST /v1/mapari (save manual): re-rezolva EXCLUSIV canalul API (batch_id IS NULL).
|
||||
# T7/R1 INCHIS: salvarea unei mapari NU re-queues randuri din batches de import
|
||||
# (cross-batch / cross-feed). Batches de import sunt re-rezolvate doar la commit explicit.
|
||||
# Salvarea unei mapari NU re-queues randuri din batches de import (cross-batch /
|
||||
# cross-feed). Batches de import sunt re-rezolvate doar la commit explicit.
|
||||
rows = conn.execute(
|
||||
"SELECT id, payload_json FROM submissions "
|
||||
"WHERE status='needs_mapping' AND account_id=? AND batch_id IS NULL",
|
||||
@@ -649,7 +868,7 @@ def reresolve_account(conn, account_id: int | None, batch_id: int | None = None)
|
||||
content["prestatii"] = resolved
|
||||
payload_json = json.dumps(content, ensure_ascii=False)
|
||||
|
||||
# US-010: telemetrie pentru itemii rezolvati prin regula text.
|
||||
# Telemetrie pentru itemii rezolvati prin regula text.
|
||||
_emite_text_rule_hits(conn, acct, r["id"], resolved)
|
||||
|
||||
if unmapped:
|
||||
@@ -660,18 +879,8 @@ def reresolve_account(conn, account_id: int | None, batch_id: int | None = None)
|
||||
stats["still_blocked"] += 1
|
||||
continue
|
||||
|
||||
# T6/OV-1: verifica auto_send inainte de re-queuing
|
||||
if has_no_auto_send(resolved, mapping_meta):
|
||||
conn.execute(
|
||||
"UPDATE submissions SET payload_json=?, rar_error=?, updated_at=datetime('now') WHERE id=?",
|
||||
(
|
||||
payload_json,
|
||||
json.dumps({"auto_send": "cod mapat cu auto_send=0; review manual inainte de trimitere"}, ensure_ascii=False),
|
||||
r["id"],
|
||||
),
|
||||
)
|
||||
stats["review_manual"] += 1
|
||||
continue
|
||||
# US-001 (PRD 5.11): ramura auto_send eliminata din reresolve.
|
||||
# Un cod rezolvat -> queued direct (review_manual ramane 0).
|
||||
|
||||
errors = validate_prezentare(content)
|
||||
if errors:
|
||||
|
||||
@@ -1,9 +1,8 @@
|
||||
"""Modele Pydantic pentru suprafata API.
|
||||
|
||||
ATENTIE: validarea completa (regex VIN ^[A-HJ-NPR-Z0-9]{17}$, nrInmatriculare,
|
||||
dataPrestatie ∈ [2024-12-01, azi] TZ Bucuresti, R-ODO/I-ODO -> odometruInitial
|
||||
obligatoriu, odometruInitial <= odometruFinal, normalizare strip/upper) este
|
||||
**T3** — aici sunt doar formele de baza. Vezi plan.md sect. 2 + roadmap T3.
|
||||
Aici sunt doar formele de baza + normalizare strip/upper. Validarea completa de
|
||||
continut (regex VIN, interval data, R-ODO/I-ODO -> odometruInitial, ordine
|
||||
odometru) este in app.validation.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -20,7 +19,7 @@ class RarCredentials(BaseModel):
|
||||
|
||||
|
||||
class PrestatieItem(BaseModel):
|
||||
"""O operatie de declarat. Contract hibrid (decis 2026-06-15):
|
||||
"""O operatie de declarat. Contract hibrid:
|
||||
|
||||
ROAAUTO poate trimite FIE `cod_prestatie` (cod RAR direct, ex. OE-1), FIE
|
||||
`cod_op_service` (cod intern ROAAUTO) + `denumire` — pe care gateway-ul le
|
||||
@@ -55,7 +54,7 @@ class PrezentareIn(BaseModel):
|
||||
Pydantic doar NORMALIZEAZA aici (strip/upper pe vin/nrInm). Validarea de
|
||||
continut (regex VIN, interval data, R-ODO/I-ODO, odometru) e in
|
||||
app.validation.validate_prezentare si NU resping cererea — marcheaza
|
||||
`needs_data` (plan.md sect. 3).
|
||||
`needs_data`.
|
||||
"""
|
||||
|
||||
vin: str
|
||||
@@ -102,12 +101,12 @@ class SubmissionResult(BaseModel):
|
||||
status: str
|
||||
id_prezentare: int | None = None
|
||||
deduped: bool = False # True daca idempotency a intors un submission existent
|
||||
# US-012 (decizie /autoplan #19): camp ADITIV. True cand un rand `error` cu aceeasi
|
||||
# cheie de continut a fost RE-ACTIVAT (re-clasificat + creds actualizate) la resubmit.
|
||||
# `deduped` pastreaza semantica actuala (clientii vechi care testeaza `deduped` nu se sparg).
|
||||
# Camp aditiv. True cand un rand `error` cu aceeasi cheie de continut a fost
|
||||
# RE-ACTIVAT (re-clasificat + creds actualizate) la resubmit. `deduped` pastreaza
|
||||
# semantica actuala (clientii vechi care testeaza `deduped` nu se sparg).
|
||||
reactivated: bool = False
|
||||
# Raspuns ONEST pentru randuri blocate (PRD 5.7): orice status != 'queued' isi
|
||||
# expune motivul, ca integratorul sa nu trateze un needs_data/needs_mapping drept succes.
|
||||
# Raspuns ONEST pentru randuri blocate: orice status != 'queued' isi expune
|
||||
# motivul, ca integratorul sa nu trateze un needs_data/needs_mapping drept succes.
|
||||
# erori = validare de continut (needs_data), 3 niveluri [{field, cod, problema, cauza, fix, message}].
|
||||
# Pe ramura on_unmapped_error='error' pastreaza COD_NEMAPAT (compat).
|
||||
# nemapate = coduri fara mapare RAR (needs_mapping / respins), 3 niveluri + cod_op_service/denumire.
|
||||
@@ -122,7 +121,7 @@ class PrezentariResponse(BaseModel):
|
||||
|
||||
|
||||
class ValidarePrezentariRequest(BaseModel):
|
||||
"""Body pentru POST /v1/prezentari/valideaza — dry-run fara enqueue (PRD 5.2)."""
|
||||
"""Body pentru POST /v1/prezentari/valideaza — dry-run fara enqueue."""
|
||||
|
||||
rar_credentials: RarCredentials | None = None
|
||||
prezentari: list[PrezentareIn] = Field(..., min_length=1)
|
||||
|
||||
@@ -1,12 +1,11 @@
|
||||
"""Logger structurat central (PRD 5.6 US-003).
|
||||
"""Logger structurat central.
|
||||
|
||||
Singurul punct prin care se emit evenimente de aplicatie: garanteaza format,
|
||||
redactare si dublul canal (app_events in DB + log text rotativ) consistente si
|
||||
imposibil de ocolit. Best-effort ca `notify_signup`: o cadere a jurnalului NU
|
||||
doboara cererea/worker-ul.
|
||||
imposibil de ocolit. Best-effort: o cadere a jurnalului NU doboara cererea/worker-ul.
|
||||
|
||||
Redactare la SCRIERE (nu la afisare): toate valorile trec prin `redact_pii`
|
||||
(creds/token mascate integral, VIN/nr partial) inainte de persistare (US-007).
|
||||
(creds/token mascate integral, VIN/nr partial) inainte de persistare.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -22,8 +21,8 @@ from .config import get_settings
|
||||
from .db import get_connection, insert_app_event
|
||||
from .security import redact_pii, scrub_text
|
||||
|
||||
# request_id al cererii curente (US-002). Setat de middleware-ul HTTP; disponibil
|
||||
# in handlerul de erori (US-001) si aici, fara a polua semnaturile de functii.
|
||||
# request_id al cererii curente. Setat de middleware-ul HTTP; disponibil in
|
||||
# handlerul de erori si aici, fara a polua semnaturile de functii.
|
||||
request_id_var: contextvars.ContextVar[str | None] = contextvars.ContextVar(
|
||||
"request_id", default=None
|
||||
)
|
||||
@@ -31,7 +30,7 @@ request_id_var: contextvars.ContextVar[str | None] = contextvars.ContextVar(
|
||||
_LEVELS = {"DEBUG": 10, "INFO": 20, "WARNING": 30, "WARN": 30, "ERROR": 40, "CRITICAL": 50}
|
||||
|
||||
# Sursa implicita a evenimentelor pentru procesul curent. API = 'api' (default);
|
||||
# worker-ul cheama set_source('worker') la pornire (T5: fisier per-proces).
|
||||
# worker-ul cheama set_source('worker') la pornire (fisier per-proces).
|
||||
_DEFAULT_SOURCE = "api"
|
||||
|
||||
_loggers: dict[str, logging.Logger] = {}
|
||||
@@ -46,9 +45,9 @@ def set_source(sursa: str) -> None:
|
||||
def _text_logger(sursa: str) -> logging.Logger:
|
||||
"""Logger cu RotatingFileHandler pe fisier per-proces (app-<sursa>.log).
|
||||
|
||||
Rotatia pe dimensiune e in aplicatie (decizie §5) — nu depindem de deploy.
|
||||
Cheia de cache include calea: la schimbarea log_dir (teste) se creeaza un logger
|
||||
nou, fara a acumula handlere duplicate pe acelasi fisier.
|
||||
Rotatia pe dimensiune e in aplicatie — nu depindem de deploy. Cheia de cache
|
||||
include calea: la schimbarea log_dir (teste) se creeaza un logger nou, fara a
|
||||
acumula handlere duplicate pe acelasi fisier.
|
||||
"""
|
||||
settings = get_settings()
|
||||
path = settings.log_dir / f"app-{sursa}.log"
|
||||
@@ -94,10 +93,10 @@ def log_event(
|
||||
) -> None:
|
||||
"""Emite un eveniment: un rand `app_events` + o linie in logul text (acelasi continut redactat).
|
||||
|
||||
- `tip`: text liber documentat (lista extensibila, decizie §5).
|
||||
- `tip`: text liber documentat (lista extensibila).
|
||||
- `nivel`: DEBUG|INFO|WARNING|ERROR|CRITICAL. Sub `AUTOPASS_LOG_LEVEL` -> ignorat.
|
||||
- `context`: metadate (submission_id, count, status...) — NU payload PII integral.
|
||||
- `conn`: reutilizeaza conexiunea apelantului pe calea fierbinte (evita contentie WAL, T4);
|
||||
- `conn`: reutilizeaza conexiunea apelantului pe calea fierbinte (evita contentie WAL);
|
||||
None -> deschide/inchide o conexiune proprie.
|
||||
Best-effort: orice exceptie e inghitita (jurnalul nu trebuie sa rupa fluxul).
|
||||
"""
|
||||
|
||||
59
app/operatii_seed.py
Normal file
59
app/operatii_seed.py
Normal file
@@ -0,0 +1,59 @@
|
||||
"""Seeder corpus operatii etichetate -> mapping_suggestions (SILVER, PRD 5.18 US-004).
|
||||
|
||||
Artefactul `app/data/operatii-etichetate.json` e produs offline de
|
||||
`tools/mapare-llm/genereaza_seed.py` (etichetare LM Studio, o singura data) si comis
|
||||
in repo. La `init_db` il incarcam in `mapping_suggestions` cu INSERT OR IGNORE, ca
|
||||
SILVER sa nu mai fie gol in productie (sugestii exact-match + corpus k-NN reale).
|
||||
|
||||
Format seed: [{denumire, denumire_normalizata, cod, is_nul, source, confidence}].
|
||||
Reutilizeaza `shared_store.seed_suggestions` (normalizeaza cheia + impune NUL->cod NULL,
|
||||
INSERT OR IGNORE). NB (F10): confirmarile UMANE stau in `shared_mappings`, NU aici —
|
||||
deci INSERT OR IGNORE pastreaza codul LLM existent la re-seed (v1 = ignore, nu upsert).
|
||||
|
||||
SUGGESTION-ONLY (invariant #13): nimic din SILVER nu intra in resolve_prestatii/load_mapping.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import json
|
||||
import os
|
||||
import sqlite3
|
||||
|
||||
from .shared_store import seed_suggestions
|
||||
|
||||
SEED_PATH = os.path.join(os.path.dirname(__file__), "data", "operatii-etichetate.json")
|
||||
|
||||
|
||||
def load_seed_file(path: str = SEED_PATH) -> list[dict]:
|
||||
"""Citeste artefactul seed. Lipsa / invalid -> [] (degradare gratioasa)."""
|
||||
if not path or not os.path.exists(path):
|
||||
return []
|
||||
try:
|
||||
with open(path, encoding="utf-8") as fh:
|
||||
data = json.load(fh)
|
||||
except (ValueError, OSError):
|
||||
return []
|
||||
return data if isinstance(data, list) else []
|
||||
|
||||
|
||||
def seed_operatii_etichetate(conn: sqlite3.Connection, path: str = SEED_PATH) -> int:
|
||||
"""Incarca seedul in mapping_suggestions (INSERT OR IGNORE). Intoarce nr. randuri inserate.
|
||||
|
||||
Mapeaza cheia seedului `cod` -> `cod_prestatie` (forma asteptata de seed_suggestions);
|
||||
`is_nul=True` forteaza cod NULL acolo. Idempotent: re-rularea nu dubleaza randuri.
|
||||
"""
|
||||
raw = load_seed_file(path)
|
||||
if not raw:
|
||||
return 0
|
||||
items = [
|
||||
{
|
||||
"denumire": e.get("denumire") or e.get("denumire_normalizata") or "",
|
||||
"cod_prestatie": e.get("cod"),
|
||||
"is_nul": bool(e.get("is_nul")),
|
||||
"source": e.get("source") or "llm_seed",
|
||||
"confidence": e.get("confidence") or 0.0,
|
||||
}
|
||||
for e in raw
|
||||
if isinstance(e, dict)
|
||||
]
|
||||
return seed_suggestions(conn, items)
|
||||
@@ -1,4 +1,4 @@
|
||||
"""Extragere payload submission -> campuri afisabile (US-003, PRD 3.5).
|
||||
"""Extragere payload submission -> campuri afisabile.
|
||||
|
||||
Helper PUR partajat intre canalul web (dashboard Trimiteri) si canalul API
|
||||
(`GET /v1/prezentari`), ca extragerea sa NU diverge intre cele doua (decizie
|
||||
@@ -115,6 +115,20 @@ def prezentare_din_payload(payload: str | dict | None) -> dict[str, str]:
|
||||
# cod_rar: exclusiv din cod_prestatie (NU fallback la cod_op_service); uppercase + strip ".0"
|
||||
cod_rar = _clean_cod_rar(item.get("cod_prestatie"))
|
||||
|
||||
# Operatia de service originala (codul intern + denumire venita prin API/import),
|
||||
# distincta de operatia RAR mapata (cod_rar).
|
||||
# Conventie goala: aceste campuri intorc "" (string gol) cand lipsesc — NU EMPTY="—".
|
||||
# Motivul: apelantul decide sa nu afiseze randul deloc (vs afisaj gol), testând `!= ""`.
|
||||
# Campurile vechi (vehicul_nr, vin, operatie etc.) pastreaza conventia EMPTY="—".
|
||||
op_service_cod = _clean_str(item.get("cod_op_service"))
|
||||
# op_service_denumire e relevant doar cand exista un cod de operatie de service;
|
||||
# altfel ar expune denumirea RAR drept op. de service, ceea ce e semantic incorect.
|
||||
op_service_denumire = _clean_str(item.get("denumire")) if op_service_cod else ""
|
||||
|
||||
# obs: text liber observatii (camp RAR, optional). Conventie goala "" (nu EMPTY).
|
||||
# US-005 PRD 5.15: obs traieste in payload_json (nu coloana separata).
|
||||
obs = _clean_str(data.get("obs"))
|
||||
|
||||
return {
|
||||
"vehicul_nr": nr or EMPTY,
|
||||
"vin": vin or EMPTY,
|
||||
@@ -124,4 +138,8 @@ def prezentare_din_payload(payload: str | dict | None) -> dict[str, str]:
|
||||
"odometru": odo or EMPTY,
|
||||
"cod": cod or EMPTY,
|
||||
"cod_rar": cod_rar or EMPTY,
|
||||
# Chei cu conventie goala "" (nu EMPTY) — vezi comentariu de mai sus
|
||||
"op_service_cod": op_service_cod,
|
||||
"op_service_denumire": op_service_denumire,
|
||||
"obs": obs,
|
||||
}
|
||||
|
||||
130
app/plans.py
Normal file
130
app/plans.py
Normal file
@@ -0,0 +1,130 @@
|
||||
"""Definitia planurilor de cont (sursa unica de adevar). Modul PUR, fara import DB/HTTP.
|
||||
|
||||
Pattern ca app/errors.py: catalog + helperi. Consumat de rutele de ingestie si dashboard.
|
||||
Nu importa DB, HTTP, sau orice alt modul intern cu efecte secundare.
|
||||
|
||||
Decizii implementare (PRD 5.17 / autoplan 2026-06-28):
|
||||
- FREE_MONTHLY_LIMIT: constanta unica (T-CEO-2), tunabila fara arqueologie de cod.
|
||||
- CONSUMED_STATUSES: decizie #20 — prestatie consumata = acceptata in coada.
|
||||
- effective_tier: `now` injectabil (decizie #2) pentru teste deterministe.
|
||||
- monthly_usage: pattern E7/5.15 (strftime localtime), `now` injectabil.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import sqlite3
|
||||
from datetime import datetime, timezone
|
||||
|
||||
# Limita lunara pentru planul Gratuit.
|
||||
# Decizie user T-CEO-2 (2026-06-28): o singura constanta, referita din PLANS.
|
||||
# Tunabila fara a modifica logica de enforcement.
|
||||
FREE_MONTHLY_LIMIT: int = 60
|
||||
|
||||
# Statusurile care consuma din cota lunara (decizie #20, 2026-06-28).
|
||||
# Prestatie consumata = acceptata in coada (queued/sending/sent), nu cele respinse/blocate.
|
||||
# Rationale: limita e pe ce trimitem la RAR, nu pe incercari esuate sau blocate.
|
||||
CONSUMED_STATUSES: tuple[str, ...] = ("queued", "sending", "sent")
|
||||
|
||||
# Sursa unica de adevar pentru planuri. Fiecare plan are:
|
||||
# label -- eticheta afisata in RO (UI, mesaje)
|
||||
# monthly_limit -- None = nelimitat; int = limita prestatii/luna
|
||||
# api_access -- True = acces import prin API (/v1/*); False = doar web dashboard
|
||||
#
|
||||
# Aliniat landing-ului comercial (PRD 5.17 US-001):
|
||||
# Gratuit: 60/luna, fara API
|
||||
# Standard: nelimitat, fara API
|
||||
# Pro: nelimitat, cu API
|
||||
# Premium: nelimitat, cu API (suport dedicat)
|
||||
PLANS: dict[str, dict] = {
|
||||
"free": {
|
||||
"label": "Gratuit",
|
||||
"monthly_limit": FREE_MONTHLY_LIMIT,
|
||||
"api_access": False,
|
||||
},
|
||||
"standard": {
|
||||
"label": "Standard",
|
||||
"monthly_limit": None,
|
||||
"api_access": False,
|
||||
},
|
||||
"pro": {
|
||||
"label": "Pro",
|
||||
"monthly_limit": None,
|
||||
"api_access": True,
|
||||
},
|
||||
"premium": {
|
||||
"label": "Premium",
|
||||
"monthly_limit": None,
|
||||
"api_access": True,
|
||||
},
|
||||
}
|
||||
|
||||
|
||||
def effective_tier(account_row, now: datetime) -> str:
|
||||
"""Returneaza tier-ul efectiv al contului la momentul `now` (injectabil pentru determinism).
|
||||
|
||||
Daca `trial_until` e in viitor -> 'pro' (trial Pro activ).
|
||||
Altfel -> `tier`-ul de baza al contului.
|
||||
trial_until malformat/NULL -> fallback defensiv la tier de baza (nu arunca niciodata).
|
||||
|
||||
`now` TREBUIE injectat explicit (nu datetime.now() intern) — decizie #2 din autoplan.
|
||||
Suporta sqlite3.Row si dict.
|
||||
"""
|
||||
# Citire robusta: suporta sqlite3.Row (IndexError pe key absent) si dict (KeyError)
|
||||
try:
|
||||
tier = account_row["tier"]
|
||||
except (KeyError, IndexError, TypeError):
|
||||
tier = "free"
|
||||
try:
|
||||
trial_until_str = account_row["trial_until"]
|
||||
except (KeyError, IndexError, TypeError):
|
||||
trial_until_str = None
|
||||
|
||||
# Fallback defensiv la 'free' daca tier e None/gol
|
||||
if not tier:
|
||||
tier = "free"
|
||||
|
||||
if not trial_until_str:
|
||||
return tier
|
||||
|
||||
try:
|
||||
# Parseaza trial_until; stocam ca "YYYY-MM-DD HH:MM:SS" (UTC implicit) sau ISO
|
||||
tu = datetime.fromisoformat(trial_until_str.strip().replace(" ", "T"))
|
||||
# Daca fara timezone -> assume UTC (cum stocam in DB)
|
||||
if tu.tzinfo is None:
|
||||
tu = tu.replace(tzinfo=timezone.utc)
|
||||
# Normalizeaza `now` la aware daca e naive
|
||||
now_cmp = now
|
||||
if now_cmp.tzinfo is None:
|
||||
now_cmp = now_cmp.replace(tzinfo=timezone.utc)
|
||||
if tu > now_cmp:
|
||||
return "pro"
|
||||
except (ValueError, AttributeError, TypeError):
|
||||
pass # malformat -> fallback defensiv la tier de baza
|
||||
|
||||
return tier
|
||||
|
||||
|
||||
def monthly_usage(conn: sqlite3.Connection, account_id: int, now: datetime) -> int:
|
||||
"""Numara prestatiile contului acceptate in coada in luna calendaristica curenta.
|
||||
|
||||
Definitia 'luna curenta': strftime('%Y-%m', created_at, 'localtime') corespunde
|
||||
lunii lui `now` (acelasi pattern ca E7/5.15 din routes.py — consistent cu 'localtime').
|
||||
`now` injectabil pentru teste deterministe. Scoped strict pe account_id.
|
||||
created_at NULL/malformat -> exclus defensiv (nu arunca niciodata).
|
||||
|
||||
NOTA: containerul are /etc/localtime=UTC, deci 'localtime' = UTC in mediul de test.
|
||||
Testele de granita construiesc timestamp-uri relative la luna curenta calculata cu
|
||||
acelasi 'localtime', nu valori absolute care presupun +2/+3h.
|
||||
"""
|
||||
# Formatam `now` ca string SQLite si folosim acelasi modificator 'localtime' ca routes.py
|
||||
now_str = now.strftime("%Y-%m-%d %H:%M:%S")
|
||||
placeholders = ",".join("?" * len(CONSUMED_STATUSES))
|
||||
row = conn.execute(
|
||||
f"SELECT COUNT(*) AS n FROM submissions "
|
||||
f"WHERE account_id = ? "
|
||||
f" AND status IN ({placeholders}) "
|
||||
f" AND created_at IS NOT NULL "
|
||||
f" AND strftime('%Y-%m', created_at, 'localtime') = strftime('%Y-%m', ?, 'localtime')",
|
||||
(account_id, *CONSUMED_STATUSES, now_str),
|
||||
).fetchone()
|
||||
return int(row["n"]) if row else 0
|
||||
91
app/rar_env.py
Normal file
91
app/rar_env.py
Normal file
@@ -0,0 +1,91 @@
|
||||
"""Medii RAR per cont (PRD 5.20): disponibilitate + default efectiv.
|
||||
|
||||
Sursa UNICA de adevar pentru REQ-DISP / REQ-DEFAULT: vizibilitatea selector/toggle
|
||||
in UI, validarea tintei in API si decizia worker-ului citesc TOATE de aici, ca sa
|
||||
decida identic.
|
||||
|
||||
Un mediu ('test'|'prod') e *disponibil* pentru un cont daca e activat (bifa) SI are
|
||||
credentiale (slot per-mediu non-gol). Din disponibilitate decurge tot UX-ul:
|
||||
- 0 medii -> nicio tinta; trimiterea web e blocata, API cade pe ancora globala.
|
||||
- 1 mediu -> tinta implicita (acel mediu), fara selector.
|
||||
- 2 medii -> selector la import + toggle in statusbar + alegere in API.
|
||||
|
||||
Functii PURE (fara DB) peste un rand de cont (sqlite3.Row sau dict). Helperele cu
|
||||
`conn` incarca randul si deleaga.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import sqlite3
|
||||
from typing import Any
|
||||
|
||||
VALID_ENVS: tuple[str, str] = ("test", "prod")
|
||||
|
||||
|
||||
def _field(account: Any, key: str, default: Any = None) -> Any:
|
||||
"""Citire toleranta a unui camp de cont (dict sau sqlite3.Row, camp posibil absent)."""
|
||||
if account is None:
|
||||
return default
|
||||
if isinstance(account, dict):
|
||||
return account.get(key, default)
|
||||
try:
|
||||
return account[key] # sqlite3.Row
|
||||
except (IndexError, KeyError):
|
||||
return default
|
||||
|
||||
|
||||
def _are_creds(account: Any, env: str) -> bool:
|
||||
creds = _field(account, f"rar_creds_{env}_enc", None)
|
||||
return bool(creds and str(creds).strip())
|
||||
|
||||
|
||||
def _enabled(account: Any, env: str) -> bool:
|
||||
return int(_field(account, f"rar_{env}_enabled", 0) or 0) == 1
|
||||
|
||||
|
||||
def medii_disponibile(account: Any) -> list[str]:
|
||||
"""Subset din ('test','prod') = activat AND creds prezente. Ordine stabila test<prod."""
|
||||
return [env for env in VALID_ENVS if _enabled(account, env) and _are_creds(account, env)]
|
||||
|
||||
|
||||
def rar_env_efectiv(account: Any) -> str | None:
|
||||
"""Mediul tinta implicit al contului (REQ-DEFAULT).
|
||||
|
||||
Mereu unul din mediile disponibile: default-ul contului daca inca e disponibil,
|
||||
altfel singurul disponibil; daca 0 disponibile -> None (nicio tinta).
|
||||
"""
|
||||
disp = medii_disponibile(account)
|
||||
if not disp:
|
||||
return None
|
||||
default = _field(account, "rar_env_default", "prod")
|
||||
if default in disp:
|
||||
return default
|
||||
return disp[0]
|
||||
|
||||
|
||||
# --------------------------------------------------------------------------- #
|
||||
# Helpere cu conexiune #
|
||||
# --------------------------------------------------------------------------- #
|
||||
|
||||
_ACCOUNT_ENV_COLS = (
|
||||
"id, rar_test_enabled, rar_prod_enabled, "
|
||||
"rar_creds_test_enc, rar_creds_prod_enc, rar_env_default"
|
||||
)
|
||||
|
||||
|
||||
def load_account_env(conn: sqlite3.Connection, account_id: int) -> sqlite3.Row | None:
|
||||
"""Randul de cont cu exact coloanele de mediu (pentru medii_disponibile/rar_env_efectiv)."""
|
||||
from .mapping import account_or_default
|
||||
|
||||
return conn.execute(
|
||||
f"SELECT {_ACCOUNT_ENV_COLS} FROM accounts WHERE id=?",
|
||||
(account_or_default(account_id),),
|
||||
).fetchone()
|
||||
|
||||
|
||||
def medii_disponibile_cont(conn: sqlite3.Connection, account_id: int) -> list[str]:
|
||||
return medii_disponibile(load_account_env(conn, account_id))
|
||||
|
||||
|
||||
def rar_env_efectiv_cont(conn: sqlite3.Connection, account_id: int) -> str | None:
|
||||
return rar_env_efectiv(load_account_env(conn, account_id))
|
||||
@@ -10,6 +10,7 @@ CREATE TABLE IF NOT EXISTS accounts (
|
||||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
name TEXT NOT NULL,
|
||||
cui TEXT,
|
||||
email TEXT, -- email canonic de contact al firmei (US-001, PRD 5.12); nullable pt. conturi legacy
|
||||
active INTEGER NOT NULL DEFAULT 1, -- lifecycle cont (3.1); gate „in asteptare" consumat de 3.3
|
||||
-- Stare de ciclu de viata explicita (5.5). Superset al lui `active`: mentinem invariantul
|
||||
-- active=1 <=> status='active' (vezi accounts.set_status / set_active). Worker gate-uieste pe status.
|
||||
@@ -18,12 +19,36 @@ CREATE TABLE IF NOT EXISTS accounts (
|
||||
-- vezi accounts.delete_account — randul ramane doar pentru audit).
|
||||
status TEXT NOT NULL DEFAULT 'active'
|
||||
CHECK (status IN ('pending','active','blocked','archived','deleted')),
|
||||
rar_creds_enc TEXT, -- creds RAR criptate (Fernet) durabile per-cont (D4/Eng#1)
|
||||
rar_creds_enc TEXT, -- LEGACY (PRD 5.20 US-013 dropeaza coloana): creds RAR durabile env-less
|
||||
-- Medii RAR per cont (PRD 5.20 US-001). Fiecare mediu = bifa de activare + slot creds.
|
||||
-- medii_disponibile = enabled AND creds prezente (app/rar_env.py). Cont client nou =
|
||||
-- Productie on / Testare off (clientii declara real); contul operator se pune manual pe Testare.
|
||||
rar_test_enabled INTEGER NOT NULL DEFAULT 0 CHECK (rar_test_enabled IN (0, 1)),
|
||||
rar_prod_enabled INTEGER NOT NULL DEFAULT 1 CHECK (rar_prod_enabled IN (0, 1)),
|
||||
rar_creds_test_enc TEXT, -- creds RAR criptate (Fernet) pentru mediul Testare
|
||||
rar_creds_prod_enc TEXT, -- creds RAR criptate (Fernet) pentru mediul Productie
|
||||
rar_env_default TEXT NOT NULL DEFAULT 'prod' CHECK (rar_env_default IN ('test', 'prod')),
|
||||
-- Comportament implicit la cod prestatie necunoscut/nemapat pe canalul API:
|
||||
-- 0 (default, non-distructiv: submission 'needs_mapping', intra in editorul de mapare) sau
|
||||
-- 1 (respinge cererea fara enqueue). Override per-cerere via PrezentareRequest.on_unmapped_error.
|
||||
on_unmapped_error_default INTEGER NOT NULL DEFAULT 0
|
||||
CHECK (on_unmapped_error_default IN (0, 1)),
|
||||
-- Plan de cont (5.17). Tier de baza al contului (admin aloca manual via CLI set-tier).
|
||||
-- trial_until: daca != NULL si > now -> effective_tier() intoarce 'pro' (trial Pro activ).
|
||||
-- Cont nou primeste tier='free' + trial_until=now+30z via create_account.
|
||||
-- Contul implicit id=1 (dev) primeste DEFAULT 'free' + trial_until=NULL (fara trial).
|
||||
tier TEXT NOT NULL DEFAULT 'free'
|
||||
CHECK (tier IN ('free','standard','pro','premium')),
|
||||
trial_until TEXT, -- ISO datetime UTC sau NULL; nullable
|
||||
-- Planul CERUT de client la signup (separat de `tier`). NU acorda drepturi:
|
||||
-- `tier` ramane sursa unica de adevar pentru gate-ul API (require_api_access) si volum.
|
||||
-- Folosit la integrarea platilor: client cere plan -> plateste -> admin/webhook urca `tier`
|
||||
-- -> API se deblocheaza. NULL = necunoscut (cont creat via CLI / inainte de coloana).
|
||||
requested_plan TEXT
|
||||
CHECK (requested_plan IS NULL OR requested_plan IN ('free','standard','pro','premium')),
|
||||
-- Marca temporala a acceptarii Termenilor + politicii de confidentialitate (GDPR, L.142).
|
||||
-- Setata la signup (proba de consimtamant). NULL = cont fara flux de consimtamant (CLI/legacy).
|
||||
consent_at TEXT,
|
||||
created_at TEXT NOT NULL DEFAULT (datetime('now'))
|
||||
);
|
||||
-- Un CUI = un cont (cand e prezent). NULL ramane distinct nativ in SQLite -> conturi
|
||||
@@ -71,6 +96,10 @@ CREATE TABLE IF NOT EXISTS submissions (
|
||||
status TEXT NOT NULL DEFAULT 'queued'
|
||||
CHECK (status IN ('queued','sending','sent','needs_mapping','needs_data','error')),
|
||||
payload_json TEXT NOT NULL,
|
||||
-- Mediul RAR tinta al acestei trimiteri (PRD 5.20 US-001). DEFAULT 'test' e doar plasa
|
||||
-- pentru randuri net-noi care nu seteaza explicit; fiecare INSERT (API/import/web) seteaza
|
||||
-- rar_env explicit. Backfill din AUTOPASS_RAR_ENV global la migrare (NU lasa pe DEFAULT).
|
||||
rar_env TEXT NOT NULL DEFAULT 'test' CHECK (rar_env IN ('test', 'prod')),
|
||||
rar_creds_enc TEXT, -- creds RAR criptate (Fernet), sterse dupa primul login reusit
|
||||
rar_status_code INTEGER,
|
||||
rar_error TEXT,
|
||||
@@ -125,6 +154,7 @@ CREATE TABLE IF NOT EXISTS import_rows (
|
||||
row_index INTEGER NOT NULL,
|
||||
raw_json TEXT NOT NULL, -- PII criptat (Fernet, ca submissions)
|
||||
override_json TEXT, -- patch CANONIC editat in preview, criptat Fernet (3.6, Approach B); NULL = fara editare
|
||||
reviewed INTEGER NOT NULL DEFAULT 0, -- US-007 (PRD 5.12): 0=neconfirmat, 1=confirmat de operator; NU intra in payload/idempotenta
|
||||
resolved_status TEXT NOT NULL DEFAULT 'pending'
|
||||
CHECK (resolved_status IN (
|
||||
'pending','ok','needs_mapping','needs_data',
|
||||
@@ -198,6 +228,42 @@ CREATE TABLE IF NOT EXISTS operation_text_rules (
|
||||
);
|
||||
CREATE INDEX IF NOT EXISTS idx_text_rules_account ON operation_text_rules(account_id);
|
||||
|
||||
-- Sugestii de mapare (strat SILVER, L14-S3 PRD 5.14).
|
||||
-- Etichete LLM/embedding — bootstrap; citita DOAR de suggest_codes/pending_unmapped,
|
||||
-- NICIODATA de load_mapping/resolve_prestatii (separare structurala #13).
|
||||
-- Cheia = denumire normalizata (fara diacritice, uppercase, spatii colapsate).
|
||||
-- is_nul=1: non-operatie (ITP, discount, nr. inmatriculare) -> suprima (#4), cod NULL.
|
||||
-- INSERT OR IGNORE la re-seed: nu suprascrie randuri existente (#2).
|
||||
CREATE TABLE IF NOT EXISTS mapping_suggestions (
|
||||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
denumire_normalizata TEXT NOT NULL UNIQUE,
|
||||
cod_prestatie TEXT, -- NULL cand is_nul=1 (supresie)
|
||||
is_nul INTEGER NOT NULL DEFAULT 0 CHECK (is_nul IN (0, 1)),
|
||||
source TEXT NOT NULL, -- 'llm', 'embedding', etc. (#5)
|
||||
confidence REAL NOT NULL DEFAULT 0.0 CHECK (confidence >= 0.0 AND confidence <= 1.0),
|
||||
created_at TEXT NOT NULL DEFAULT (datetime('now')),
|
||||
updated_at TEXT NOT NULL DEFAULT (datetime('now'))
|
||||
);
|
||||
CREATE INDEX IF NOT EXISTS idx_mapping_suggestions_cod
|
||||
ON mapping_suggestions(cod_prestatie) WHERE cod_prestatie IS NOT NULL;
|
||||
|
||||
-- Mapari validate de oameni (strat GOLD partajat cross-account, L14-S3 PRD 5.14).
|
||||
-- Confirmarile umane din ORICE cont contribuie la acest store (#8).
|
||||
-- cross-account = suggestion-only (pre-completeaza editorul, F-A/#11), NU auto-send.
|
||||
-- Auto-send DOAR din operations_mapping (GOLD propriu per-cont, Eng-F2).
|
||||
-- Cheia = denumire_normalizata (NU cod_op_service: spatii de chei diferite, #14).
|
||||
CREATE TABLE IF NOT EXISTS shared_mappings (
|
||||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
denumire_normalizata TEXT NOT NULL UNIQUE,
|
||||
cod_prestatie TEXT NOT NULL, -- cod RAR valid (GOLD = validat de om)
|
||||
source TEXT NOT NULL DEFAULT 'human', -- 'human', 'human_import' (#5)
|
||||
provenance TEXT, -- detalii: cont, email, batch (#5)
|
||||
confidence REAL NOT NULL DEFAULT 1.0,
|
||||
confirmations INTEGER NOT NULL DEFAULT 1, -- contor confirmari din orice cont
|
||||
created_at TEXT NOT NULL DEFAULT (datetime('now')),
|
||||
updated_at TEXT NOT NULL DEFAULT (datetime('now'))
|
||||
);
|
||||
|
||||
-- Heartbeat worker (un singur rand, id=1). /healthz citeste de aici.
|
||||
CREATE TABLE IF NOT EXISTS worker_heartbeat (
|
||||
id INTEGER PRIMARY KEY CHECK (id = 1),
|
||||
|
||||
@@ -40,7 +40,7 @@ SENSITIVE_KEYS = frozenset(
|
||||
|
||||
|
||||
# Chei al caror continut e PII de identificare vehicul/proprietar: se logheaza DOAR
|
||||
# partial (ultimele 4), niciodata integral (PRD 5.6 US-007, L.142/GDPR).
|
||||
# partial (ultimele 4), niciodata integral (L.142/GDPR).
|
||||
PII_PARTIAL_KEYS = frozenset({"vin", "nr_inmatriculare", "nr", "numar"})
|
||||
|
||||
|
||||
|
||||
142
app/shared_store.py
Normal file
142
app/shared_store.py
Normal file
@@ -0,0 +1,142 @@
|
||||
"""Store partajat pentru sugestii (SILVER) si mapari validate de oameni (GOLD cross-account).
|
||||
|
||||
Straturi (L14-S3, PRD 5.14):
|
||||
- mapping_suggestions (SILVER): sugestii LLM/embedding, citite DOAR de suggest/pending_unmapped,
|
||||
NICIODATA de load_mapping/resolve_prestatii (separare structurala #13).
|
||||
- shared_mappings (GOLD partajat): mapari validate de om din orice cont; pot pre-completa
|
||||
editorul (suggestion-only cross-account, F-A/#11); auto-send DOAR GOLD propriu
|
||||
(operations_mapping per-cont, Eng-F2).
|
||||
|
||||
Invariante:
|
||||
- INSERT OR IGNORE la seed: nu suprascrie randuri existente (#2).
|
||||
- NUL = is_nul=1, cod_prestatie NULL; NU se promoveaza la cod RAR (#4).
|
||||
- source/confidence pe fiecare rand (provenienta + rollback batch model prost, #5).
|
||||
- Wiring in resolve_prestatii/load_mapping vine in L14-S6; modulul de fata e API pur.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import sqlite3
|
||||
from typing import Any
|
||||
|
||||
from .mapping import normalize_for_match
|
||||
|
||||
|
||||
def seed_suggestions(
|
||||
conn: sqlite3.Connection,
|
||||
items: list[dict[str, Any]],
|
||||
) -> int:
|
||||
"""Insereaza sugestii in mapping_suggestions (SILVER). INSERT OR IGNORE.
|
||||
|
||||
Nu suprascrie randuri deja existente (#2): re-rularea seeder-ului e sigura.
|
||||
Fiecare item trebuie sa contina:
|
||||
- 'denumire': str — text brut (se normalizeaza intern cu normalize_for_match)
|
||||
- 'source': str — 'llm', 'embedding', etc.
|
||||
Optional:
|
||||
- 'cod_prestatie': str | None — ignorat cand is_nul=True
|
||||
- 'is_nul': bool — True pt non-operatii (supresie, #4); cod_prestatie stocat NULL
|
||||
- 'confidence': float — 0..1 (default 0.0)
|
||||
|
||||
Returneaza numarul de randuri inserate efectiv (0 daca toate existau deja).
|
||||
"""
|
||||
inserted = 0
|
||||
for item in items:
|
||||
den_norm = normalize_for_match(item.get("denumire") or "")
|
||||
if not den_norm:
|
||||
continue
|
||||
is_nul = 1 if item.get("is_nul") else 0
|
||||
# NUL -> cod NULL obligatoriu (supresie stricta, #4)
|
||||
# Normalizeaza INAINTE de truthiness: un cod whitespace-only (" ") sau
|
||||
# ne-string trebuie sa devina NULL, nu '' (altfel rand non-NUL cu cod gol).
|
||||
cod = None
|
||||
if not is_nul:
|
||||
raw_cod = str(item.get("cod_prestatie") or "").strip().upper()
|
||||
cod = raw_cod or None
|
||||
source = str(item.get("source") or "llm")
|
||||
confidence = float(item.get("confidence") or 0.0)
|
||||
cur = conn.execute(
|
||||
"""
|
||||
INSERT OR IGNORE INTO mapping_suggestions
|
||||
(denumire_normalizata, cod_prestatie, is_nul, source, confidence)
|
||||
VALUES (?, ?, ?, ?, ?)
|
||||
""",
|
||||
(den_norm, cod, is_nul, source, confidence),
|
||||
)
|
||||
inserted += cur.rowcount
|
||||
return inserted
|
||||
|
||||
|
||||
def lookup_suggestion(
|
||||
conn: sqlite3.Connection,
|
||||
denumire: str,
|
||||
) -> sqlite3.Row | None:
|
||||
"""Cauta sugestie SILVER dupa denumire normalizata.
|
||||
|
||||
Returneaza randul din mapping_suggestions sau None daca nu exista.
|
||||
NOTA: apelantul trebuie sa verifice is_nul inainte de a folosi cod_prestatie.
|
||||
"""
|
||||
den_norm = normalize_for_match(denumire)
|
||||
if not den_norm:
|
||||
return None
|
||||
return conn.execute(
|
||||
"SELECT * FROM mapping_suggestions WHERE denumire_normalizata = ?",
|
||||
(den_norm,),
|
||||
).fetchone()
|
||||
|
||||
|
||||
def lookup_shared_gold(
|
||||
conn: sqlite3.Connection,
|
||||
denumire: str,
|
||||
) -> sqlite3.Row | None:
|
||||
"""Cauta mapare GOLD partajata dupa denumire normalizata.
|
||||
|
||||
Returneaza randul din shared_mappings sau None daca nu exista.
|
||||
NOTA (F-A/#11): acest GOLD partajat e suggestion-only cross-account;
|
||||
auto-send vine DOAR din operations_mapping (GOLD propriu per-cont).
|
||||
"""
|
||||
den_norm = normalize_for_match(denumire)
|
||||
if not den_norm:
|
||||
return None
|
||||
return conn.execute(
|
||||
"SELECT * FROM shared_mappings WHERE denumire_normalizata = ?",
|
||||
(den_norm,),
|
||||
).fetchone()
|
||||
|
||||
|
||||
def record_human_validation(
|
||||
conn: sqlite3.Connection,
|
||||
denumire: str,
|
||||
cod_prestatie: str,
|
||||
*,
|
||||
source: str = "human",
|
||||
provenance: str | None = None,
|
||||
confidence: float = 1.0,
|
||||
) -> None:
|
||||
"""Inregistreaza o mapare validata de om in GOLD partajat (shared_mappings).
|
||||
|
||||
Daca denumirea exista deja: incrementeaza confirmations + actualizeaza updated_at.
|
||||
Daca nu exista: insert nou cu confirmations=1.
|
||||
|
||||
Apelat la confirmarea umana a unei mapari (din editorul needs_mapping).
|
||||
Wiring efectiv vine in L14-S6 (dupa 5.15); aceasta functie e API-ul store.
|
||||
|
||||
NOTA: NU intra in operations_mapping (GOLD per-cont) — acela e gestionat
|
||||
separat de editorul existent. Ambele pot coexista.
|
||||
"""
|
||||
den_norm = normalize_for_match(denumire)
|
||||
if not den_norm:
|
||||
return
|
||||
cod = (cod_prestatie or "").strip().upper()
|
||||
if not cod:
|
||||
return
|
||||
conn.execute(
|
||||
"""
|
||||
INSERT INTO shared_mappings
|
||||
(denumire_normalizata, cod_prestatie, source, provenance, confidence, confirmations)
|
||||
VALUES (?, ?, ?, ?, ?, 1)
|
||||
ON CONFLICT(denumire_normalizata) DO UPDATE SET
|
||||
confirmations = confirmations + 1,
|
||||
updated_at = datetime('now')
|
||||
""",
|
||||
(den_norm, cod, source, provenance, confidence),
|
||||
)
|
||||
@@ -1,19 +1,19 @@
|
||||
"""Lifecycle trimiteri blocate: sterge / re-pune in coada (PRD 5.6 US-009).
|
||||
"""Lifecycle trimiteri blocate: sterge / re-pune in coada.
|
||||
|
||||
Inchide lacuna descoperita live: un rand `error` (creds RAR gresite) ramane altfel
|
||||
permanent si nereparabil. Aceste helpere adauga DOUA tranzitii controlate —
|
||||
stergere de randuri ne-sent si `blocate -> queued` (re-clasificat) — fara a atinge
|
||||
logica de trimitere a worker-ului.
|
||||
Inchide lacuna: un rand `error` (creds RAR gresite) ar ramane altfel permanent si
|
||||
nereparabil. Aceste helpere adauga DOUA tranzitii controlate — stergere de randuri
|
||||
ne-sent si `blocate -> queued` (re-clasificat) — fara a atinge logica de trimitere
|
||||
a worker-ului.
|
||||
|
||||
Invariante (decizii §2 + /autoplan #20):
|
||||
Invariante:
|
||||
- Opereaza DOAR pe `error`/`needs_data`/`needs_mapping`. `sent` (dovada de trimitere
|
||||
la RAR, audit) si `sending` (lease worker in zbor) sunt INTERZISE.
|
||||
- Scope-ul (apartenenta la cont) se evalueaza INAINTEA starii: un rand inexistent SAU
|
||||
al altui cont -> SubmissionNotFound (404, nu confirmam existenta, B3). Doar pe randuri
|
||||
al altui cont -> SubmissionNotFound (404, nu confirmam existenta). Doar pe randuri
|
||||
proprii in stare gresita -> SubmissionStateConflict (409).
|
||||
- Ambele emit eveniment in jurnal (US-003): `submission_sters` / `submission_repus`.
|
||||
- Ambele emit eveniment in jurnal: `submission_sters` / `submission_repus`.
|
||||
|
||||
Functii cu `conn` (persistenta). Apelate din API (US-010) si din web (US-011).
|
||||
Functii cu `conn` (persistenta).
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -80,7 +80,7 @@ def requeue_submission(conn, account_id: int, sid: int) -> dict:
|
||||
|
||||
`error -> queued` (cand continutul e valid) sau ramane `needs_data`/`needs_mapping`
|
||||
daca clasificarea o cere. Reseteaza retry_count/next_attempt_at/sending_since si
|
||||
CURATA `purge_after` (randul redevine activ, nu mai e candidat la purjare — US-013).
|
||||
CURATA `purge_after` (randul redevine activ, nu mai e candidat la purjare).
|
||||
Ridica SubmissionNotFound / SubmissionStateConflict. Intoarce
|
||||
{"submission_id", "status_anterior", "status_nou"}.
|
||||
"""
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
"""Helper-e utilizatori web (email + parola scrypt). US-001 PRD 3.3.
|
||||
"""Helper-e utilizatori web (email + parola scrypt).
|
||||
|
||||
Parola NICIODATA stocata in clar. Fiecare user are un salt per-user generat cu
|
||||
secrets.token_bytes(16). Parametrii scrypt stocati ca eticheta de versiune pentru
|
||||
migrare cost viitoare (C9).
|
||||
migrare cost viitoare.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
@@ -98,7 +98,7 @@ def set_admin(conn: sqlite3.Connection, account_id: int, is_admin: bool = True)
|
||||
"""Seteaza/sterge rolul admin pe toti userii contului dat.
|
||||
|
||||
Ridica ValueError daca contul nu exista.
|
||||
Daca contul exista dar nu are useri, e no-op silentios (confom spec US-010).
|
||||
Daca contul exista dar nu are useri, e no-op silentios.
|
||||
"""
|
||||
acct = conn.execute("SELECT 1 FROM accounts WHERE id=?", (account_id,)).fetchone()
|
||||
if not acct:
|
||||
@@ -119,7 +119,7 @@ def is_account_admin(conn: sqlite3.Connection, account_id: int) -> bool:
|
||||
|
||||
|
||||
def list_admin_emails(conn: sqlite3.Connection) -> list[str]:
|
||||
"""Returneaza emailurile tuturor userilor cu is_admin=1 (folosit de US-012)."""
|
||||
"""Returneaza emailurile tuturor userilor cu is_admin=1."""
|
||||
rows = conn.execute(
|
||||
"SELECT email FROM users WHERE is_admin=1"
|
||||
).fetchall()
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
"""Panou admin web /admin. US-011 PRD 3.3b.
|
||||
"""Panou admin web /admin.
|
||||
|
||||
Rute:
|
||||
GET /admin — listeaza conturi in asteptare + active (require_admin)
|
||||
@@ -8,6 +8,7 @@ Rute:
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from datetime import datetime, timedelta, timezone
|
||||
from pathlib import Path
|
||||
|
||||
from fastapi import APIRouter, Form, Request
|
||||
@@ -15,12 +16,42 @@ from fastapi.responses import HTMLResponse, RedirectResponse
|
||||
from fastapi.templating import Jinja2Templates
|
||||
|
||||
from .. import __version__
|
||||
from ..accounts import list_accounts, set_active, set_status, delete_account
|
||||
from ..accounts import account_is_complete, list_accounts, set_active, set_status, set_tier, set_trial, delete_account
|
||||
from ..config import get_settings
|
||||
from ..db import get_connection
|
||||
from ..plans import PLANS, effective_tier
|
||||
from ..web.csrf import get_csrf_token, verify_csrf
|
||||
from ..web.session import require_admin
|
||||
|
||||
|
||||
def _plan_label(code: str | None) -> str:
|
||||
"""Eticheta RO a unui cod de plan (din PLANS). None/necunoscut -> '—'."""
|
||||
if not code:
|
||||
return "—"
|
||||
plan = PLANS.get(code)
|
||||
return plan["label"] if plan else code
|
||||
|
||||
|
||||
def _trial_zile_ramase(trial_until_str: str | None, now: datetime) -> int | None:
|
||||
"""Zile ramase din trial (rotunjit in sus), sau None daca nu e trial activ/malformat.
|
||||
|
||||
Acelasi parsing tolerant ca plans.effective_tier (UTC implicit pe valori naive).
|
||||
"""
|
||||
if not trial_until_str:
|
||||
return None
|
||||
try:
|
||||
tu = datetime.fromisoformat(trial_until_str.strip().replace(" ", "T"))
|
||||
if tu.tzinfo is None:
|
||||
tu = tu.replace(tzinfo=timezone.utc)
|
||||
now_cmp = now if now.tzinfo else now.replace(tzinfo=timezone.utc)
|
||||
secunde = (tu - now_cmp).total_seconds()
|
||||
if secunde <= 0:
|
||||
return None
|
||||
# Rotunjire in sus la zile (o fractie de zi ramasa = inca 1 zi afisata).
|
||||
return int(secunde // 86400) + (1 if secunde % 86400 else 0)
|
||||
except (ValueError, AttributeError, TypeError):
|
||||
return None
|
||||
|
||||
router = APIRouter()
|
||||
_TMPL = Jinja2Templates(directory=str(Path(__file__).resolve().parent / "templates"))
|
||||
|
||||
@@ -47,9 +78,20 @@ def _render_admin(request: Request, conn, *, error: str | None = None, status_co
|
||||
"""Randeaza pagina admin.html cu lista de conturi si optional un mesaj de eroare."""
|
||||
accounts = list_accounts(conn)
|
||||
emails = _emails_by_account(conn)
|
||||
now = datetime.now(timezone.utc)
|
||||
for acct in accounts:
|
||||
# Computa is_complete INAINTE de a suprascrie accounts.email cu emailul de login al userului
|
||||
acct["is_complete"] = account_is_complete(acct)
|
||||
acct["email"] = emails.get(acct["id"])
|
||||
# Grupare pe STARE (5.5), nu pe `active`: altfel conturile arhivate/blocate (active=0)
|
||||
# Plan EFECTIV (ce are contul acum): trial Pro activ ridica `free` la `pro`.
|
||||
# `tier` ramane sursa de adevar pentru drepturi; `requested_plan` e doar intentia de la signup.
|
||||
eff = effective_tier(acct, now)
|
||||
acct["tier_label"] = _plan_label(acct.get("tier")) # tier de baza (post-trial)
|
||||
acct["tier_efectiv_label"] = _plan_label(eff) # plan efectiv ACUM
|
||||
acct["trial_activ"] = eff != (acct.get("tier") or "free")
|
||||
acct["trial_zile"] = _trial_zile_ramase(acct.get("trial_until"), now)
|
||||
acct["requested_plan_label"] = _plan_label(acct.get("requested_plan"))
|
||||
# Grupare pe STARE, nu pe `active`: altfel conturile arhivate/blocate (active=0)
|
||||
# ar cadea gresit sub "in asteptare". 'deleted' e deja exclus din list_accounts.
|
||||
pending = [a for a in accounts if a["status"] == "pending" and a["id"] != 1]
|
||||
active = [a for a in accounts if a["status"] == "active" and a["id"] != 1]
|
||||
@@ -79,12 +121,18 @@ async def admin_get(request: Request):
|
||||
|
||||
|
||||
def _apply_lifecycle(conn, ids: list[int], action: str) -> None:
|
||||
"""Aplica un verb de ciclu de viata (5.5) pe o lista de conturi. Conturile protejate
|
||||
"""Aplica un verb de ciclu de viata pe o lista de conturi. Conturile protejate
|
||||
(id=1) sau inexistente ridica ValueError din helperi -> sarite (nu opresc bulk-ul).
|
||||
`action`: activate | block | archive | delete."""
|
||||
for aid in ids:
|
||||
try:
|
||||
if action == "activate":
|
||||
# Gate US-002: nu activam conturi fara identitate completa (companie+email+CUI)
|
||||
acct_row = conn.execute(
|
||||
"SELECT id, name, cui, email FROM accounts WHERE id=?", (aid,)
|
||||
).fetchone()
|
||||
if acct_row and not account_is_complete(acct_row):
|
||||
continue # sarim activarea — contul incomplet ramane pending
|
||||
set_status(conn, aid, "active")
|
||||
elif action == "block":
|
||||
set_status(conn, aid, "blocked")
|
||||
@@ -97,7 +145,7 @@ def _apply_lifecycle(conn, ids: list[int], action: str) -> None:
|
||||
|
||||
|
||||
def _lifecycle_route(request: Request, account_id: list[int], csrf_token: str, action: str):
|
||||
"""Corp comun pentru rutele de ciclu de viata (5.5): auth + CSRF + aplica verbul (bulk) + PRG.
|
||||
"""Corp comun pentru rutele de ciclu de viata: auth + CSRF + aplica verbul (bulk) + PRG.
|
||||
Evita 4 handlere copy-paste care difera doar prin verb."""
|
||||
require_admin(request)
|
||||
verify_csrf(request, csrf_token)
|
||||
@@ -138,6 +186,73 @@ async def admin_delete(request: Request, account_id: list[int] = Form(...),
|
||||
return _lifecycle_route(request, account_id, csrf_token, "delete")
|
||||
|
||||
|
||||
@router.post("/admin/set-tier", response_class=HTMLResponse)
|
||||
async def admin_set_tier(
|
||||
request: Request,
|
||||
account_id: int = Form(...),
|
||||
tier: str = Form(...),
|
||||
csrf_token: str = Form(default=""),
|
||||
):
|
||||
"""Schimba planul (tier) unui cont din panou. require_admin + CSRF, PRG 303.
|
||||
|
||||
Reuseaza accounts.set_tier (valideaza tier-ul, protejeaza id=1, logheaza schimbarea).
|
||||
INCHEIE trial-ul (trial_until=NULL): alocarea manuala = plan real de-acum, cu efect
|
||||
imediat — altfel trial-ul Pro universal (30z la signup) ar masca alegerea pana la
|
||||
expirare (decizie user 2026-06-29). Tier invalid / cont protejat -> re-randare cu eroare.
|
||||
"""
|
||||
require_admin(request)
|
||||
verify_csrf(request, csrf_token)
|
||||
|
||||
conn = get_connection()
|
||||
try:
|
||||
try:
|
||||
# trial_until=None: alocarea manuala incheie trial-ul si aplica tier-ul ales acum.
|
||||
set_tier(conn, account_id, tier, trial_until=None)
|
||||
conn.commit()
|
||||
except ValueError as exc:
|
||||
return _render_admin(request, conn, error=str(exc), status_code=422)
|
||||
finally:
|
||||
conn.close()
|
||||
|
||||
return RedirectResponse("/admin", status_code=303)
|
||||
|
||||
|
||||
@router.post("/admin/set-trial", response_class=HTMLResponse)
|
||||
async def admin_set_trial(
|
||||
request: Request,
|
||||
account_id: int = Form(...),
|
||||
trial_days: int = Form(...),
|
||||
csrf_token: str = Form(default=""),
|
||||
):
|
||||
"""Acorda/prelungeste un trial Pro de N zile (de la acum), fara a schimba tier-ul de baza.
|
||||
|
||||
require_admin + CSRF, PRG 303. Reuseaza accounts.set_trial (protejeaza id=1, logheaza).
|
||||
trial_days <= 0 sau peste plafon -> re-randare panou cu eroare (422). Plafon defensiv 3650z.
|
||||
"""
|
||||
require_admin(request)
|
||||
verify_csrf(request, csrf_token)
|
||||
|
||||
conn = get_connection()
|
||||
try:
|
||||
if trial_days <= 0 or trial_days > 3650:
|
||||
return _render_admin(
|
||||
request, conn,
|
||||
error="Numarul de zile pentru trial trebuie sa fie intre 1 si 3650.",
|
||||
status_code=422,
|
||||
)
|
||||
try:
|
||||
now = datetime.now(timezone.utc)
|
||||
trial_until = (now + timedelta(days=trial_days)).strftime("%Y-%m-%d %H:%M:%S")
|
||||
set_trial(conn, account_id, trial_until)
|
||||
conn.commit()
|
||||
except ValueError as exc:
|
||||
return _render_admin(request, conn, error=str(exc), status_code=422)
|
||||
finally:
|
||||
conn.close()
|
||||
|
||||
return RedirectResponse("/admin", status_code=303)
|
||||
|
||||
|
||||
@router.post("/admin/deactivate", response_class=HTMLResponse)
|
||||
async def admin_deactivate(
|
||||
request: Request,
|
||||
|
||||
@@ -1,7 +1,8 @@
|
||||
"""Rute autentificare web: /signup (US-003), /login + /logout (US-004). PRD 3.3."""
|
||||
"""Rute autentificare web: /signup, /login, /logout."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from datetime import datetime, timezone
|
||||
from pathlib import Path
|
||||
|
||||
from fastapi import APIRouter, Form, Request
|
||||
@@ -9,7 +10,7 @@ from fastapi.responses import HTMLResponse, RedirectResponse
|
||||
from fastapi.templating import Jinja2Templates
|
||||
|
||||
from .. import __version__
|
||||
from ..accounts import create_account
|
||||
from ..accounts import VALID_TIERS, create_account
|
||||
from ..auth import create_api_key
|
||||
from ..config import get_settings
|
||||
from ..db import get_connection
|
||||
@@ -47,10 +48,18 @@ async def signup_post(
|
||||
cui: str = Form(default=""),
|
||||
email: str = Form(default=""),
|
||||
parola: str = Form(default=""),
|
||||
plan: str = Form(default=""),
|
||||
consent: str = Form(default=""),
|
||||
csrf_token: str = Form(default=""),
|
||||
):
|
||||
verify_csrf(request, csrf_token)
|
||||
|
||||
# Planul CERUT (intentie, nu drept): pastram doar valori valide; orice altceva -> 'free'.
|
||||
# `tier`-ul real ramane 'free' la creare; planul ales se onoreaza dupa plata (admin/webhook).
|
||||
requested_plan = plan.strip().lower() if plan else ""
|
||||
if requested_plan not in VALID_TIERS:
|
||||
requested_plan = "free"
|
||||
|
||||
settings = get_settings()
|
||||
ip = request.client.host if request.client else "unknown"
|
||||
if not check_rate_limit(ip, settings.signup_rate_max, settings.signup_rate_window_s):
|
||||
@@ -58,7 +67,7 @@ async def signup_post(
|
||||
request,
|
||||
csrf_token=get_csrf_token(request),
|
||||
error=_RATE_MSG,
|
||||
name=name, cui=cui, email=email,
|
||||
name=name, cui=cui, email=email, plan=requested_plan,
|
||||
), status_code=429)
|
||||
|
||||
if len(parola) < _PASSWORD_MIN:
|
||||
@@ -66,9 +75,29 @@ async def signup_post(
|
||||
request,
|
||||
csrf_token=get_csrf_token(request),
|
||||
error=f"Parola trebuie sa aiba cel putin {_PASSWORD_MIN} caractere.",
|
||||
name=name, cui=cui, email=email,
|
||||
name=name, cui=cui, email=email, plan=requested_plan,
|
||||
), status_code=422)
|
||||
|
||||
# CUI obligatoriu la signup (US-001, PRD 5.12)
|
||||
cui_norm = cui.strip().upper() if cui else ""
|
||||
if not cui_norm:
|
||||
return _TMPL.TemplateResponse(request, "signup.html", _ctx(
|
||||
request,
|
||||
csrf_token=get_csrf_token(request),
|
||||
error="CUI-ul firmei este obligatoriu.",
|
||||
name=name, cui=cui, email=email, plan=requested_plan,
|
||||
), status_code=422)
|
||||
|
||||
# Consimtamant Termeni + GDPR obligatoriu (proba). Checkbox bifat -> valoare ne-goala.
|
||||
if not (consent and consent.strip()):
|
||||
return _TMPL.TemplateResponse(request, "signup.html", _ctx(
|
||||
request,
|
||||
csrf_token=get_csrf_token(request),
|
||||
error="Trebuie sa accepti Termenii si prelucrarea datelor (GDPR) pentru a crea cont.",
|
||||
name=name, cui=cui, email=email, plan=requested_plan,
|
||||
), status_code=422)
|
||||
consent_at = datetime.now(timezone.utc).strftime("%Y-%m-%d %H:%M:%S")
|
||||
|
||||
# Bootstrap admin: count_admins se citeste INAUNTRUL tranzactiei BEGIN IMMEDIATE,
|
||||
# astfel lock-ul RESERVED serializeaza scriitorii si al doilea signup vede count==1.
|
||||
conn = get_connection()
|
||||
@@ -76,17 +105,53 @@ async def signup_post(
|
||||
conn.execute("BEGIN IMMEDIATE")
|
||||
try:
|
||||
is_first = count_admins(conn) == 0
|
||||
account_id = create_account(conn, name, cui.strip() or None, active=False)
|
||||
account_id = create_account(
|
||||
conn, name, cui=cui_norm, email=email, active=False,
|
||||
requested_plan=requested_plan, consent_at=consent_at,
|
||||
)
|
||||
user_id = create_user(conn, account_id, email, parola, is_admin=is_first)
|
||||
api_key = create_api_key(conn, account_id)
|
||||
conn.execute("COMMIT")
|
||||
except ValueError as exc:
|
||||
conn.execute("ROLLBACK")
|
||||
exc_msg = str(exc)
|
||||
# Ordinea conteaza: verifica EMAIL inainte de CUI (ambele contin 'deja folosit').
|
||||
# create_user ridica exact "email deja folosit"; create_account ridica "CUI X e deja folosit".
|
||||
if "email deja folosit" in exc_msg:
|
||||
# Email duplicat -> mesaj specific emailului (T3, D#14-email)
|
||||
error_msg = (
|
||||
"Acest email este deja folosit. "
|
||||
"Daca ai deja cont, autentifica-te."
|
||||
)
|
||||
elif "deja folosit" in exc_msg or "IntegrityError" in exc_msg:
|
||||
# CUI duplicat -> mesaj prietenos, NU mesajul tehnic cu 'activate --account' (T3, D#14)
|
||||
settings = get_settings()
|
||||
if settings.support_email:
|
||||
error_msg = (
|
||||
f"Aceasta firma (CUI {cui_norm}) e deja inregistrata. "
|
||||
f"Cere accesul de la administratorul contului sau contacteaza suportul: "
|
||||
f"{settings.support_email}"
|
||||
)
|
||||
else:
|
||||
error_msg = (
|
||||
f"Aceasta firma (CUI {cui_norm}) e deja inregistrata. "
|
||||
f"Cere accesul de la administratorul contului."
|
||||
)
|
||||
else:
|
||||
error_msg = exc_msg
|
||||
return _TMPL.TemplateResponse(request, "signup.html", _ctx(
|
||||
request,
|
||||
csrf_token=get_csrf_token(request),
|
||||
error=error_msg,
|
||||
name=name, cui=cui, email=email, plan=requested_plan,
|
||||
), status_code=422)
|
||||
except Exception as exc:
|
||||
conn.execute("ROLLBACK")
|
||||
return _TMPL.TemplateResponse(request, "signup.html", _ctx(
|
||||
request,
|
||||
csrf_token=get_csrf_token(request),
|
||||
error=str(exc),
|
||||
name=name, cui=cui, email=email,
|
||||
name=name, cui=cui, email=email, plan=requested_plan,
|
||||
), status_code=422)
|
||||
finally:
|
||||
conn.close()
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
"""CSRF token per-sesiune + validare. US-009 PRD 3.3.
|
||||
"""CSRF token per-sesiune + validare.
|
||||
|
||||
Contract pentru rutele POST web:
|
||||
- Formulare HTML includ: <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
|
||||
@@ -1,9 +1,6 @@
|
||||
"""
|
||||
labels.py — traducere stari tehnice in text uman + clasa CSS (US-001, PRD 3.4).
|
||||
"""Traducere stari tehnice in text uman + clasa CSS.
|
||||
|
||||
Functii pure: fara DB, fara request. Usor de testat unitar si de importat in template-uri.
|
||||
|
||||
Sursa de adevar pentru texte: tabelul din PRD 3.4 §3 US-001.
|
||||
"""
|
||||
|
||||
import json
|
||||
@@ -59,7 +56,7 @@ STARI_SUBMISSION: dict[str, Eticheta] = {
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Etichete scurte (pill) pentru coloana Stare din tabelul de trimiteri (US-006)
|
||||
# Etichete scurte (pill) pentru coloana Stare din tabelul de trimiteri
|
||||
# Dict propriu — NU element in tuple Eticheta (ar rupe template-urile care
|
||||
# despacheteaza 3 elemente). eticheta_stare ramane neatinsa.
|
||||
# ---------------------------------------------------------------------------
|
||||
@@ -155,7 +152,7 @@ def eticheta_rar(stare: str) -> Eticheta:
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Format data RAR (US-001, PRD 3.5)
|
||||
# Format data RAR
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
def format_data_rar(raw: object) -> str:
|
||||
@@ -181,7 +178,7 @@ def format_data_rar(raw: object) -> str:
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Motiv uman din rar_error (US-004, PRD 3.5)
|
||||
# Motiv uman din rar_error
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
def motiv_uman(status: str, rar_error: object) -> str:
|
||||
@@ -231,7 +228,7 @@ def motiv_uman(status: str, rar_error: object) -> str:
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# parse_erori — transforma rar_error in lista 3-niveluri (US-006, PRD 5.4)
|
||||
# parse_erori — transforma rar_error in lista 3-niveluri
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
def parse_erori(rar_error: object) -> list[dict]:
|
||||
@@ -275,7 +272,7 @@ def parse_erori(rar_error: object) -> list[dict]:
|
||||
"cauza": e.get("cauza") or e.get("message") or "",
|
||||
"fix": e.get("fix") or "",
|
||||
"field": e.get("field"),
|
||||
# Cod BRUT de catalog (ex. RAR_EROARE_SERVER) — DOAR pentru modal (US-001/R1).
|
||||
# Cod BRUT de catalog (ex. RAR_EROARE_SERVER) — DOAR pentru modal.
|
||||
"cod": e.get("cod"),
|
||||
})
|
||||
else:
|
||||
@@ -305,7 +302,7 @@ def parse_erori(rar_error: object) -> list[dict]:
|
||||
"cauza": data.get("cauza") or "",
|
||||
"fix": data.get("fix") or "",
|
||||
"field": data.get("field"),
|
||||
# Cod BRUT de catalog (ex. COD_NEMAPAT) — DOAR pentru modal (US-001/R1).
|
||||
# Cod BRUT de catalog (ex. COD_NEMAPAT) — DOAR pentru modal.
|
||||
"cod": data.get("cod"),
|
||||
}]
|
||||
# Dict vechi: unmapped
|
||||
@@ -330,6 +327,80 @@ def parse_erori(rar_error: object) -> list[dict]:
|
||||
return [{"problema": str(data)[:200], "cauza": "", "fix": "", "field": None}]
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Etichete stari preview import (vocabular DIFERIT de starile de submission)
|
||||
#
|
||||
# Starile de preview (ok/needs_review/already_sent/duplicate_in_file) NU
|
||||
# exista in STARI_SUBMISSION — reutilizarea directa a eticheta_stare/eticheta_scurta
|
||||
# ridica KeyError. Acest map este sursa de adevar pentru stratul de adaptare din
|
||||
# _web_compute_preview (routes.py) si pentru template (_preview_rand.html).
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
STARI_PREVIEW: dict[str, tuple[str, str]] = {
|
||||
"ok": ("Gata de trimis", "s-ok"),
|
||||
"needs_review": ("Verifica valori", "s-needs_review"),
|
||||
"needs_mapping": ("Cod RAR lipsa", "s-needs_mapping"),
|
||||
"needs_data": ("Date incomplete", "s-needs_data"),
|
||||
"already_sent": ("Deja trimis", "s-already_sent"),
|
||||
"duplicate_in_file": ("Duplicat in fisier", "s-duplicate_in_file"),
|
||||
}
|
||||
|
||||
|
||||
def nota_umana_preview(status: str, errors: list, flags: list) -> str:
|
||||
"""Formateaza mesajul uman pentru coloana Note din tabelul de preview import.
|
||||
|
||||
Primeste ``errors`` ca lista Python (nu JSON string) — NU pasa la motiv_uman
|
||||
sau parse_erori care asteapta un JSON string si ar produce repr Python brut
|
||||
prin fallback ``raw[:160]`` (bug documentat in PRD 5.11 US-003).
|
||||
|
||||
Logica de prioritate:
|
||||
- already_sent / duplicate_in_file -> "" (template le afiseaza separat)
|
||||
- needs_mapping -> unmapped INAINTE de flags (codul lipsa e motivul real)
|
||||
- flags non-goale -> primul flag (needs_review: data ambigua etc.)
|
||||
- errors cu cheie "unmapped" -> "Cod RAR lipsa pentru: COD1, COD2"
|
||||
- errors cu field+message (needs_data) -> primul mesaj de validare
|
||||
- altceva -> ""
|
||||
|
||||
Fara exceptii. Trunchiat la 200 caractere.
|
||||
"""
|
||||
if status in ("already_sent", "duplicate_in_file"):
|
||||
return ""
|
||||
# needs_mapping: codul RAR lipseste — prioritizeaza 'unmapped' inaintea flags,
|
||||
# altfel un rand cu si un flag (ex. VIN numeric) ar afisa textul flag-ului
|
||||
# si ascunde motivul real (cod lipsa).
|
||||
if status == "needs_mapping":
|
||||
for e in errors:
|
||||
if not isinstance(e, dict):
|
||||
continue
|
||||
if "unmapped" in e:
|
||||
ops = e.get("unmapped") or []
|
||||
coduri = ", ".join(
|
||||
o.get("cod_op_service", "") for o in ops if isinstance(o, dict)
|
||||
)
|
||||
return ("Cod RAR lipsa pentru: " + coduri if coduri else "Cod RAR lipsa")
|
||||
if flags:
|
||||
return str(flags[0])[:200]
|
||||
for e in errors:
|
||||
if not isinstance(e, dict):
|
||||
continue
|
||||
if "unmapped" in e:
|
||||
ops = e.get("unmapped") or []
|
||||
coduri = ", ".join(
|
||||
o.get("cod_op_service", "") for o in ops if isinstance(o, dict)
|
||||
)
|
||||
return (f"Cod RAR lipsa pentru: {coduri}" if coduri else "Cod RAR lipsa")
|
||||
msg = (
|
||||
e.get("message")
|
||||
or e.get("msg")
|
||||
or e.get("problema")
|
||||
or e.get("cauza")
|
||||
or ""
|
||||
)
|
||||
if msg:
|
||||
return str(msg)[:200]
|
||||
return ""
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Constante auxiliare (microcopy fix, fara logica)
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
"""Middleware HTTP: request_id per cerere (PRD 5.6 US-002).
|
||||
"""Middleware HTTP: request_id per cerere.
|
||||
|
||||
Fiecare raspuns primeste un header `X-Request-ID` (generat daca clientul nu trimite
|
||||
unul). Pe durata cererii, id-ul e disponibil prin `observ.request_id_var` (contextvar)
|
||||
in handlerul de erori (US-001) si in `log_event` (US-003) — fara a polua semnaturile.
|
||||
in handlerul de erori si in `log_event` — fara a polua semnaturile.
|
||||
|
||||
Format opac, fara PII: `secrets.token_hex(8)` (16 hex). Daca clientul trimite un
|
||||
`X-Request-ID`, il pastram (corelare end-to-end), dar il scurtam defensiv (max 64).
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
"""Rate-limit in-proces cu fereastra glisanta. US-009 PRD 3.3 C5.
|
||||
"""Rate-limit in-proces cu fereastra glisanta.
|
||||
|
||||
Fara dependinta externa. Folosit de POST /signup (US-003) cu cheia = IP client.
|
||||
Fara dependinta externa. Folosit de POST /signup cu cheia = IP client.
|
||||
Configurabil prin AUTOPASS_signup_rate_max / AUTOPASS_signup_rate_window_s (config.py).
|
||||
"""
|
||||
|
||||
|
||||
2182
app/web/routes.py
2182
app/web/routes.py
File diff suppressed because it is too large
Load Diff
@@ -1,6 +1,6 @@
|
||||
"""Helper-e sesiune web. US-002 PRD 3.3.
|
||||
"""Helper-e sesiune web.
|
||||
|
||||
Mecanism require_login (C11): NU un dependency FastAPI care intoarce RedirectResponse
|
||||
Mecanism require_login: NU un dependency FastAPI care intoarce RedirectResponse
|
||||
(acela nu scurtcircuiteaza handler-ul — FastAPI continua executia). In schimb:
|
||||
- require_login() RIDICA LoginRequired
|
||||
- app.main inregistreaza @app.exception_handler(LoginRequired) care intoarce
|
||||
@@ -31,7 +31,7 @@ def current_account(request: Request) -> int | None:
|
||||
|
||||
|
||||
def current_user_id(request: Request) -> int | None:
|
||||
"""user_id din sesiune sau None (C19: leaga import_attestations.confirmed_by)."""
|
||||
"""user_id din sesiune sau None (leaga import_attestations.confirmed_by)."""
|
||||
val = request.session.get("user_id")
|
||||
return int(val) if val is not None else None
|
||||
|
||||
@@ -88,7 +88,7 @@ def require_admin(request: Request) -> int:
|
||||
|
||||
|
||||
def set_session(request: Request, account_id: int, user_id: int) -> None:
|
||||
"""Seteaza sesiunea dupa login. Curata mai intai (C3 anti-fixare sesiune)."""
|
||||
"""Seteaza sesiunea dupa login. Curata mai intai (anti-fixare sesiune)."""
|
||||
request.session.clear()
|
||||
request.session["account_id"] = account_id
|
||||
request.session["user_id"] = user_id
|
||||
|
||||
BIN
app/web/static/fonts/IBMPlexMono-Regular-latin-ext.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexMono-Regular-latin-ext.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexMono-Regular-latin.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexMono-Regular-latin.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexSans-Bold-latin-ext.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexSans-Bold-latin-ext.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexSans-Bold-latin.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexSans-Bold-latin.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexSans-Medium-latin-ext.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexSans-Medium-latin-ext.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexSans-Medium-latin.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexSans-Medium-latin.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexSans-Regular-latin-ext.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexSans-Regular-latin-ext.woff2
Normal file
Binary file not shown.
BIN
app/web/static/fonts/IBMPlexSans-Regular-latin.woff2
Normal file
BIN
app/web/static/fonts/IBMPlexSans-Regular-latin.woff2
Normal file
Binary file not shown.
BIN
app/web/static/romfast_logo.png
Normal file
BIN
app/web/static/romfast_logo.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 26 KiB |
@@ -1,7 +1,29 @@
|
||||
<div id="acasa-section">
|
||||
|
||||
{# === Centru de greutate: bara de upload (importul e operatia principala) === #}
|
||||
{% include '_upload.html' %}
|
||||
{# === Banner ne-blocant: cont incomplet (US-002) ===
|
||||
Apare cand accounts.name / email / CUI sunt necompletate (conturi legacy sau create din CLI).
|
||||
NU blocheaza importul sau uploadul — doar orienteaza operatorul sa completeze datele.
|
||||
Dispare automat dupa ce contul devine complet (re-render la urmatoarea navigare/reload).
|
||||
#}
|
||||
{% if cont_incomplet %}
|
||||
<div class="card banner" style="border-color:var(--warn); background:color-mix(in srgb, var(--warn) 12%, var(--card)); margin-bottom:14px; padding:10px 14px; font-size:13px;">
|
||||
<strong>Completeaza datele firmei (email / CUI).</strong>
|
||||
Contul tau nu are inca email de contact si CUI configurate.
|
||||
<a href="/?tab=cont" style="margin-left:6px;">Completeaza acum →</a>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === Container colapsabil: stepper + upload intr-un singur element <details> (US-006).
|
||||
Serverul seteaza atributul `open` din are_trimiteri:
|
||||
are_trimiteri=False (first-run) → open (importul e vizibil imediat, fara JS)
|
||||
are_trimiteri=True (returning) → colapsat (nu ocupa ecranul, dar e accesibil la click)
|
||||
Degradare fara JS: corecta pe ambele ramuri.
|
||||
In timpul fluxului (mapcoloane/preview), HTMX face swap pe #import-section (descendentul
|
||||
intern) → <details> ramane neatins → containerul ramane deschis intre pasi. === #}
|
||||
<details id="import-details"{% if not are_trimiteri %} open{% endif %}>
|
||||
<summary>+ Importa fisier (XLSX / CSV)</summary>
|
||||
{% include '_upload.html' %}
|
||||
</details>
|
||||
|
||||
{# === Subordonat: primii pasi pe un singur rand compact === #}
|
||||
{% set toti_esentiali = are_creds and are_trimiteri %}
|
||||
@@ -44,14 +66,14 @@
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# US-001 (5.5): randul "Ajutor" (wayfinding Mapari/Coduri RAR) eliminat — navigarea
|
||||
traieste in tab-bar (Mapari) si in meniul de cont (Nomenclator etc.). #}
|
||||
|
||||
{# === Sectiunea Trimiteri ("Trimiterile tale"), permanenta sub upload (US-003).
|
||||
Suprimata la first-run (zero trimiteri): bara de upload acopera deja CTA-ul,
|
||||
iar empty-state-ul tabelului ar fi redundant (US-004 / D-5.1). === #}
|
||||
{# Sectiunea Trimiteri, permanenta sub upload.
|
||||
La first-run (zero trimiteri), randam un placeholder <section> gol/ascuns — necesar
|
||||
ca OOB swap-ul de la confirma sa gaseasca tinta valida in DOM si sa injecteze
|
||||
_coada.html fara reload complet. Fara placeholder, HTMX ignora silentios OOB-ul. #}
|
||||
{% if are_trimiteri %}
|
||||
{% include '_coada.html' %}
|
||||
{% else %}
|
||||
<section id="trimiteri-section" hidden></section>
|
||||
{% endif %}
|
||||
|
||||
</div>
|
||||
|
||||
234
app/web/templates/_chips_prestatii.html
Normal file
234
app/web/templates/_chips_prestatii.html
Normal file
@@ -0,0 +1,234 @@
|
||||
{#
|
||||
_chips_prestatii.html — sectiunea de prestatii chips (E4, server-driven via /form-chips).
|
||||
|
||||
Re-randata de endpoint-ul /form-chips la fiecare add/remove de chip.
|
||||
Inclusa si din _form_editare.html pentru randarea initiala.
|
||||
|
||||
Starea chip-urilor traieste in input-uri hidden din form (NU in DB mid-edit).
|
||||
Fiecare operatie are un picker propriu cand e nemapata (E4 binding op<->cod).
|
||||
Reveal odometru initial semnalat prin data-has-r-odo="true" si chip-warn pe R-ODO/I-ODO.
|
||||
|
||||
Context vars (toate cu defaults):
|
||||
prestatii_chips — list of {cod_prestatie, cod_op_service, denumire}
|
||||
nomenclator_rar — list of {cod_prestatie, nume_prestatie} pentru picker
|
||||
has_r_odo — True daca orice chip e R-ODO sau I-ODO (server-computed)
|
||||
form_chips_url — URL pentru HTMX; default '/form-chips'
|
||||
chips_section_id — ID div (default 'chips-section')
|
||||
csrf_token — CSRF (trecut prin hx-include din form parinte)
|
||||
#}
|
||||
{% set _chips_url = form_chips_url or '/form-chips' %}
|
||||
{% set _sec_id = chips_section_id or 'chips-section' %}
|
||||
{% set _chips = prestatii_chips or [] %}
|
||||
{% set _has_ops = _chips | selectattr('cod_op_service') | list | length > 0 %}
|
||||
{# US-009: chips_submission_id e setat din _detaliu_ctx cand chips sunt randate in modalul de detaliu.
|
||||
Lipseste cand _chips_prestatii.html e rerandat via /form-chips (stateless, fara submission). #}
|
||||
{% set _sub_id = chips_submission_id if chips_submission_id is defined else none %}
|
||||
|
||||
<div id="{{ _sec_id }}" data-has-r-odo="{{ 'true' if has_r_odo else 'false' }}"
|
||||
aria-live="polite" aria-label="Prestatii cod RAR">
|
||||
|
||||
{# ===== Input-uri hidden pentru starea curenta a chip-urilor =====
|
||||
TOATE itemele emit 3 hidden inputs (cod poate fi "" pentru unmapped).
|
||||
Paralele index-by-index: cod_prestatie[i], chip_op_service[i], chip_denumire[i].
|
||||
Filtrate la submit de post_corectie_trimitere (coduri goale = neschimbate). #}
|
||||
{% for chip in _chips %}
|
||||
<input type="hidden" name="cod_prestatie" value="{{ chip.cod_prestatie or '' }}">
|
||||
<input type="hidden" name="chip_op_service" value="{{ chip.cod_op_service or '' }}">
|
||||
<input type="hidden" name="chip_denumire" value="{{ chip.denumire or '' }}">
|
||||
{% endfor %}
|
||||
|
||||
<div class="camp-slim" style="margin-bottom:8px;">
|
||||
<label>Prestatii — cod RAR pe fiecare operatie</label>
|
||||
|
||||
{% if _has_ops %}
|
||||
{# ===== Mod operatii: UN picker PE operatie (E4 binding) ===== #}
|
||||
{% for chip in _chips %}
|
||||
{% if chip.cod_op_service %}
|
||||
{% set _is_warn = chip.cod_prestatie in ('R-ODO', 'I-ODO') %}
|
||||
{% set _nemapat = not chip.cod_prestatie %}
|
||||
<div class="op-row {% if _nemapat %}op-row-warn{% endif %}" style="margin-bottom:6px;">
|
||||
<span class="op-row-name">
|
||||
{{ chip.cod_op_service }}
|
||||
{% if chip.denumire and chip.denumire != chip.cod_op_service %}
|
||||
<span class="muted" style="font-weight:400;font-size:11px;"> — {{ chip.denumire }}</span>
|
||||
{% endif %}
|
||||
{% if _nemapat %}
|
||||
<span style="color:var(--warn);font-size:10px;font-weight:400;"> · lipsa cod</span>
|
||||
{% endif %}
|
||||
</span>
|
||||
<span style="display:flex;align-items:center;gap:8px;">
|
||||
{% if chip.cod_prestatie %}
|
||||
{# ===== Operatie mapata: chip cu × ===== #}
|
||||
<span class="chip {% if _is_warn %}chip-warn{% endif %}"
|
||||
aria-label="Prestatie {{ chip.cod_prestatie }} adaugata pentru {{ chip.cod_op_service }}">
|
||||
{{ chip.cod_prestatie }}
|
||||
<button type="button" class="chip-del"
|
||||
hx-post="{{ _chips_url }}"
|
||||
hx-include="closest form"
|
||||
hx-target="#{{ _sec_id }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-vals='{"chips_action":"remove","chips_remove_index":{{ loop.index0 }}}'
|
||||
aria-label="Sterge codul {{ chip.cod_prestatie }} pentru {{ chip.cod_op_service }}">
|
||||
×
|
||||
</button>
|
||||
</span>
|
||||
{# US-009: "salveaza ca regula op->cod" — apare doar cand submission_id e cunoscut
|
||||
(in modalul de detaliu, nu la re-randarea stateless via /form-chips).
|
||||
Reuse EXACT save_mapping + reresolve_account via endpoint dedicat.
|
||||
hx-include="closest form" propaga csrf_token din form-ul parinte. #}
|
||||
<span id="save-rule-slot-{{ loop.index0 }}" class="save-rule-slot">
|
||||
{% if _sub_id and chip.cod_op_service and chip.cod_prestatie %}
|
||||
<button type="button"
|
||||
style="font-size:10px;color:var(--muted);background:none;border:none;cursor:pointer;text-decoration:underline;padding:0;margin-left:4px;line-height:1;"
|
||||
hx-post="/trimitere/{{ _sub_id }}/salveaza-regula-chip"
|
||||
hx-include="closest form"
|
||||
hx-target="#detaliu-modal-body"
|
||||
hx-swap="innerHTML"
|
||||
hx-vals='{"salveaza_op":{{ chip.cod_op_service | tojson }},"salveaza_cod":{{ chip.cod_prestatie | tojson }}}'
|
||||
aria-label="Salveaza regula {{ chip.cod_op_service }} -> {{ chip.cod_prestatie }}">
|
||||
salveaza ca regula
|
||||
</button>
|
||||
{% endif %}
|
||||
</span>
|
||||
{% else %}
|
||||
{# ===== Operatie nemapata: picker galben cu "alege cod RAR" ===== #}
|
||||
<select name="chips_add_cod_{{ loop.index0 }}"
|
||||
id="picker-op-{{ loop.index0 }}"
|
||||
aria-label="Alege cod RAR pentru {{ chip.cod_op_service }}"
|
||||
style="min-width:160px;font-size:11px;height:26px;">
|
||||
<option value="">— alege cod RAR —</option>
|
||||
{% for n in (nomenclator_rar or []) %}
|
||||
<option value="{{ n.cod_prestatie }}">{{ n.cod_prestatie }} — {{ n.nume_prestatie }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
<button type="button"
|
||||
class="add-code"
|
||||
hx-post="{{ _chips_url }}"
|
||||
hx-include="closest form"
|
||||
hx-target="#{{ _sec_id }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-vals='{"chips_action":"add","chips_add_op_index":{{ loop.index0 }}}'
|
||||
aria-label="Adauga cod RAR pentru {{ chip.cod_op_service }}">
|
||||
+ Adauga
|
||||
</button>
|
||||
{% endif %}
|
||||
</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
|
||||
{# ===== US-005 (5.16): Chips extra + picker '+ Adauga alta operatie / cod RAR' in mod operatii ===== #}
|
||||
{# Chips extra: cod_op_service gol, cod_prestatie setat — afisate flat cu × (reuse remove_flat).
|
||||
T-7 (5.16): containerul .chips se randeaza DOAR cand exista chips extra — altfel ramanea
|
||||
un chenar gol nefinisat sub randurile de operatie. #}
|
||||
{% set _extra_chips = _chips | rejectattr('cod_op_service') | selectattr('cod_prestatie') | list %}
|
||||
{% if _extra_chips %}
|
||||
<div class="chips" role="group" aria-label="Coduri RAR suplimentare" style="margin-top:4px;">
|
||||
{% for chip in _extra_chips %}
|
||||
{% set _is_warn_extra = chip.cod_prestatie in ('R-ODO', 'I-ODO') %}
|
||||
<span class="chip {% if _is_warn_extra %}chip-warn{% endif %}"
|
||||
aria-label="Cod RAR suplimentar {{ chip.cod_prestatie }}">
|
||||
{{ chip.cod_prestatie }}
|
||||
<button type="button" class="chip-del"
|
||||
hx-post="{{ _chips_url }}"
|
||||
hx-include="closest form"
|
||||
hx-target="#{{ _sec_id }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-vals='{"chips_action":"remove_flat","chips_remove_cod":"{{ chip.cod_prestatie }}"}'
|
||||
aria-label="Sterge codul suplimentar {{ chip.cod_prestatie }}">×</button>
|
||||
</span>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endif %}
|
||||
{% if nomenclator_rar %}
|
||||
<span style="display:inline-flex;align-items:center;gap:4px;margin-top:4px;">
|
||||
<select name="chips_add_cod_flat"
|
||||
aria-label="Adauga cod RAR suplimentar"
|
||||
style="min-width:160px;font-size:11px;height:26px;border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line));border-radius:5px;background:transparent;color:var(--accent);">
|
||||
<option value="">+ Adauga alta operatie / cod RAR</option>
|
||||
{% for n in nomenclator_rar %}
|
||||
<option value="{{ n.cod_prestatie }}">{{ n.cod_prestatie }} — {{ n.nume_prestatie }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
<button type="button"
|
||||
class="add-code"
|
||||
hx-post="{{ _chips_url }}"
|
||||
hx-include="closest form"
|
||||
hx-target="#{{ _sec_id }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-vals='{"chips_action":"add_extra"}'
|
||||
aria-label="Adauga cod RAR suplimentar la trimitere">
|
||||
+
|
||||
</button>
|
||||
</span>
|
||||
{% else %}
|
||||
{# T-D1/T-E5 (5.16): empty state in mod operatii cand nomenclatorul lipseste #}
|
||||
<div class="chips-nom-gol" style="font-size:11px;color:var(--warn);padding:4px 0;margin-top:4px;">
|
||||
Nomenclator indisponibil — adaugarea de coduri suplimentare nu e posibila.
|
||||
</div>
|
||||
{% endif %}
|
||||
{% if chips_extra_error %}
|
||||
{# T-C1/T-E4 (5.16): semnal vizibil cand add_extra are select gol sau cod invalid #}
|
||||
<div class="chips-extra-error" style="font-size:11px;color:var(--err);padding:2px 0;" role="alert">
|
||||
Selecteaza un cod RAR din lista inainte de a adauga.
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{% else %}
|
||||
{# ===== Mod plat: lista de coduri libere (corectie pura, fara op_service) ===== #}
|
||||
<div class="chips" role="group" aria-label="Coduri RAR selectate">
|
||||
{% for chip in _chips %}
|
||||
{% if chip.cod_prestatie %}
|
||||
{% set _is_warn_flat = chip.cod_prestatie in ('R-ODO', 'I-ODO') %}
|
||||
<span class="chip {% if _is_warn_flat %}chip-warn{% endif %}"
|
||||
aria-label="Prestatie {{ chip.cod_prestatie }}">
|
||||
{{ chip.cod_prestatie }}
|
||||
<button type="button" class="chip-del"
|
||||
hx-post="{{ _chips_url }}"
|
||||
hx-include="closest form"
|
||||
hx-target="#{{ _sec_id }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-vals='{"chips_action":"remove_flat","chips_remove_cod":"{{ chip.cod_prestatie }}"}'
|
||||
aria-label="Sterge codul {{ chip.cod_prestatie }}">×</button>
|
||||
</span>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
{# Picker adaugare cod nou in mod plat #}
|
||||
{% if nomenclator_rar %}
|
||||
<span style="display:inline-flex;align-items:center;gap:4px;">
|
||||
<select name="chips_add_cod_flat"
|
||||
aria-label="Adauga cod RAR nou"
|
||||
style="font-size:11px;height:22px;border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line));border-radius:5px;background:transparent;color:var(--accent);">
|
||||
<option value="">+ cod</option>
|
||||
{% for n in nomenclator_rar %}
|
||||
<option value="{{ n.cod_prestatie }}">{{ n.cod_prestatie }} — {{ n.nume_prestatie }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
<button type="button"
|
||||
class="add-code"
|
||||
hx-post="{{ _chips_url }}"
|
||||
hx-include="closest form"
|
||||
hx-target="#{{ _sec_id }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-vals='{"chips_action":"add_flat"}'
|
||||
aria-label="Adauga cod RAR selectat in lista">
|
||||
+
|
||||
</button>
|
||||
</span>
|
||||
{% else %}
|
||||
{# T-D1/T-E5 (5.16): empty state in mod plat cand nomenclatorul lipseste #}
|
||||
<div class="chips-nom-gol" style="font-size:11px;color:var(--warn);padding:4px 0;">
|
||||
Nomenclator indisponibil — nu se pot adauga coduri RAR momentan.
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# Hint discret fara chips (debut) #}
|
||||
{% if not _chips %}
|
||||
<div style="font-size:10px;color:var(--muted);padding:4px 0;">
|
||||
Niciun cod RAR inca — alege din picker (sus) sau adauga prin mapare.
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
@@ -1,74 +1,98 @@
|
||||
{#
|
||||
_coada.html — repurposat in 3.6 (US-003).
|
||||
Nu mai e un tab/panou separat: e sectiunea "Trimiterile tale" inclusa pe Acasa,
|
||||
sub zona de upload. Pastreaza filtrele (US-009) si tabelul (_submissions.html); detaliul
|
||||
se deschide acum in modalul global (#modal-detaliu). Poll aliniat la 15s (anti dublu-poll, M5).
|
||||
_coada.html — sectiunea "Trimiterile tale" inclusa pe Acasa, sub zona de upload.
|
||||
Filtre + tabel (_submissions.html); detaliul se deschide in modalul global (#modal-detaliu).
|
||||
#}
|
||||
<section id="trimiteri-section" aria-labelledby="trimiteri-heading"
|
||||
style="margin-top:22px; padding-top:18px; border-top:2px solid var(--line);">
|
||||
style="margin-top:22px; padding-top:18px; border-top:2px solid var(--line);"
|
||||
{% if oob %}hx-swap-oob="outerHTML"{% endif %}>
|
||||
<div class="card">
|
||||
<div style="display:flex; align-items:center; gap:8px; flex-wrap:wrap; margin:0 0 12px;">
|
||||
<h2 id="trimiteri-heading" style="font-size:15px; margin:0;">
|
||||
Trimiterile tale
|
||||
{% if blocate_total %}
|
||||
<span class="tab-badge" title="{{ blocate_total }} necesita atentie"
|
||||
style="display:inline-flex; align-items:center; justify-content:center; min-width:18px; height:18px; margin-left:6px; padding:0 5px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;">{{ blocate_total }}</span>
|
||||
{% endif %}
|
||||
</h2>
|
||||
{# US-002 (5.16): titlul de sectiune vizibil ("Trimiterile tale") a fost eliminat —
|
||||
lista incepe direct sub filtre. Heading pastrat sr-only pentru a11y (section
|
||||
aria-labelledby). Badge-ul de atentie + export CSV stau intr-un rand discret. #}
|
||||
<h2 id="trimiteri-heading" class="sr-only">Trimiterile tale</h2>
|
||||
{% if blocate_total %}
|
||||
<div style="display:flex; align-items:center; gap:6px; flex-wrap:wrap; margin:0 0 10px;">
|
||||
<span class="tab-badge" title="{{ blocate_total }} necesita atentie"
|
||||
style="display:inline-flex; align-items:center; justify-content:center; min-width:18px; height:18px; padding:0 5px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;">{{ blocate_total }}</span>
|
||||
<span class="muted" style="font-size:var(--fs-sm);">de rezolvat</span>
|
||||
<span style="margin-left:auto; display:flex; gap:8px; flex-wrap:wrap;">
|
||||
<a class="cardlink" href="/v1/audit/export?status=sent" download>export CSV: trimise</a>
|
||||
<a class="cardlink" href="/v1/audit/export?status=all" download>toate</a>
|
||||
</span>
|
||||
</div>
|
||||
{% else %}
|
||||
<div style="display:flex; justify-content:flex-end; gap:8px; flex-wrap:wrap; margin:0 0 10px;">
|
||||
<a class="cardlink" href="/v1/audit/export?status=sent" download>export CSV: trimise</a>
|
||||
<a class="cardlink" href="/v1/audit/export?status=all" download>toate</a>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<!-- Filtre (US-009): reincarca tabelul; poll-ul re-trimite filtrul curent prin hx-include -->
|
||||
<!-- Bara de filtre: [quick-pills data STANGA] [cautare vehicul MIJLOC] [pills stare DREAPTA].
|
||||
Pill-urile de stare scriu campul hidden status si re-trimit form-ul (filtreazaStare).
|
||||
Quick-pills de data apeleaza setDataRange -> seteaza data_de/data_pana + re-submit. -->
|
||||
<form id="filtre-trimiteri"
|
||||
hx-get="/_fragments/submissions"
|
||||
hx-target="#submissions-wrap"
|
||||
hx-swap="innerHTML"
|
||||
hx-trigger="submit, change, keyup delay:400ms from:input[name='vehicul']"
|
||||
style="display:flex; gap:10px; flex-wrap:wrap; align-items:flex-end; margin-bottom:12px;">
|
||||
<div>
|
||||
<label for="f-status" class="muted" style="display:block; font-size:12px;">Stare</label>
|
||||
{# US-014/T13: status_filtru (din deep-link ?tab=acasa&status=) pre-selecteaza
|
||||
starea, iar submissions-wrap (hx-include #filtre-trimiteri) o incarca filtrat. #}
|
||||
{% set sf = status_filtru | default('') %}
|
||||
<select id="f-status" name="status">
|
||||
<option value="" {% if not sf %}selected{% endif %}>toate</option>
|
||||
<option value="queued" {% if sf == 'queued' %}selected{% endif %}>in asteptare</option>
|
||||
<option value="sent" {% if sf == 'sent' %}selected{% endif %}>declarate la RAR</option>
|
||||
<option value="needs_mapping" {% if sf == 'needs_mapping' %}selected{% endif %}>lipsa cod</option>
|
||||
<option value="needs_data" {% if sf == 'needs_data' %}selected{% endif %}>date incomplete</option>
|
||||
<option value="error" {% if sf == 'error' %}selected{% endif %}>eroare</option>
|
||||
<option value="sending" {% if sf == 'sending' %}selected{% endif %}>se trimite</option>
|
||||
</select>
|
||||
style="display:flex; gap:8px 12px; flex-wrap:wrap; align-items:center; margin-bottom:12px;">
|
||||
<input type="hidden" id="f-status" name="status" value="{{ status_filtru | default('', true) }}">
|
||||
{# Pagina curenta — actualizata prin OOB swap din _submissions.html; inclusa la reincarcari. #}
|
||||
<input type="hidden" id="f-page" name="page" value="1">
|
||||
|
||||
{# === STANGA: Quick-pills de data (preset interval) + buton Custom ===
|
||||
Azi / 7 zile / 30 zile → seteaza interval preset si submitr automat.
|
||||
Custom → dezvaluie #custom-date-fields pentru introducere manuala (fara submit automat). #}
|
||||
<div style="flex:0 0 auto; display:flex; gap:8px; align-items:center; flex-wrap:wrap;">
|
||||
<div class="pills-categorii" id="quick-date-pills">
|
||||
<button type="button" class="pill-cat pill-data" data-range="azi"
|
||||
aria-pressed="false"
|
||||
onclick="setDataRange(this,'azi')">Azi</button>
|
||||
<button type="button" class="pill-cat pill-data" data-range="7zile"
|
||||
aria-pressed="false"
|
||||
onclick="setDataRange(this,'7zile')">7 zile</button>
|
||||
<button type="button" class="pill-cat pill-data" data-range="30zile"
|
||||
aria-pressed="false"
|
||||
onclick="setDataRange(this,'30zile')">30 zile</button>
|
||||
<button type="button" class="pill-cat pill-data" data-range="custom"
|
||||
aria-pressed="false"
|
||||
onclick="setDataRange(this,'custom')">Custom</button>
|
||||
</div>
|
||||
{# Campuri de data pentru modul Custom: ascunse pana la click pe „Custom".
|
||||
type="date" (nu hidden) permite interactiunea utilizatorului.
|
||||
Campul change pe form re-incarca automat lista via hx-trigger="change". #}
|
||||
<div id="custom-date-fields"
|
||||
style="display:none; gap:4px; align-items:center; flex-wrap:wrap; font-size:13px;">
|
||||
<label for="f-data-de" class="muted" style="font-size:12px; white-space:nowrap;">De:</label>
|
||||
<input type="date" id="f-data-de" name="data_de" value=""
|
||||
style="font-size:13px; max-width:140px;">
|
||||
<label for="f-data-pana" class="muted" style="font-size:12px; white-space:nowrap;">Pana:</label>
|
||||
<input type="date" id="f-data-pana" name="data_pana" value=""
|
||||
style="font-size:13px; max-width:140px;">
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<label for="f-vehicul" class="muted" style="display:block; font-size:12px;">Vehicul (nr/VIN)</label>
|
||||
<input id="f-vehicul" type="text" name="vehicul" placeholder="ex. B123 sau VIN" style="max-width:180px;">
|
||||
|
||||
{# === MIJLOC: cautare vehicul (nr/VIN) + buton Filtreaza === #}
|
||||
<div style="display:flex; align-items:center; gap:8px; flex:1 1 auto; min-width:160px; flex-wrap:wrap;">
|
||||
<input id="f-vehicul" type="text" name="vehicul" placeholder="Vehicul (nr/VIN)"
|
||||
style="flex:1 1 auto; min-width:120px;">
|
||||
<button type="submit" style="flex:0 0 auto;">Filtreaza</button>
|
||||
</div>
|
||||
<div>
|
||||
<label for="f-data-de" class="muted" style="display:block; font-size:12px;">Data de la</label>
|
||||
<input id="f-data-de" type="date" name="data_de">
|
||||
</div>
|
||||
<div>
|
||||
<label for="f-data-pana" class="muted" style="display:block; font-size:12px;">pana la</label>
|
||||
<input id="f-data-pana" type="date" name="data_pana">
|
||||
</div>
|
||||
<button type="submit">Filtreaza</button>
|
||||
|
||||
{# === DREAPTA: pill-uri de stare cu contoare; re-randate via OOB la reincarcarea tabelului === #}
|
||||
<span id="pills-categorii" class="pills-categorii" style="margin-left:auto; flex:0 0 auto;">
|
||||
{% include '_pills.html' %}
|
||||
</span>
|
||||
</form>
|
||||
|
||||
<!-- Poll aliniat la 15s ca status-ul (M5: nu doua timere perpetue pe pagina mereu deschisa) -->
|
||||
<!-- Tabelul se reincarca la: incarcarea paginii, actiunile tale (trimiteriChanged)
|
||||
si auto-refresh periodic din poller (date noi externe). -->
|
||||
<div id="submissions-wrap"
|
||||
hx-get="/_fragments/submissions"
|
||||
hx-trigger="load, every 15s, trimiteriChanged from:body"
|
||||
hx-trigger="load, trimiteriChanged from:body, reincarcaTrimiteri"
|
||||
hx-include="#filtre-trimiteri" hx-swap="innerHTML">
|
||||
<div class="empty">se incarca…</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{# PRD 5.9 US-003: detaliul s-a mutat intr-un MODAL global (#modal-detaliu in base.html),
|
||||
in afara #submissions-wrap -> poll-ul de 15s nu-l mai atinge. Randul declanseaza
|
||||
deschiderea (hx-target=#detaliu-modal-body). Vechiul panou inert #trimitere-detaliu
|
||||
a fost eliminat (rol preluat de modal). #}
|
||||
</section>
|
||||
|
||||
@@ -1,6 +1,79 @@
|
||||
<div class="card" id="card-cont">
|
||||
<h2 style="font-size:15px; margin:0 0 16px;">Contul meu</h2>
|
||||
|
||||
<!-- Sectiunea: Plan curent (US-006 PRD 5.17) -->
|
||||
{% if plan_linie is defined %}
|
||||
<div id="sectiune-plan" style="margin-bottom:20px; padding-bottom:20px; border-bottom:1px solid var(--line);">
|
||||
<h3 style="font-size:var(--fs-sm); color:var(--muted); font-weight:500; margin:0 0 10px;
|
||||
text-transform:uppercase; letter-spacing:.04em;">Plan curent</h3>
|
||||
|
||||
<div style="font-size:var(--fs-md); font-weight:600; margin-bottom:6px;
|
||||
color:{% if plan_limita_atinsa|default(false) %}var(--err){% elif plan_warn|default(false) %}var(--warn){% else %}var(--ink){% endif %};">
|
||||
{{ plan_linie }}
|
||||
</div>
|
||||
|
||||
{% if monthly_limit_val is defined and monthly_limit_val is not none and effective_tier_name|default('') == 'free' %}
|
||||
<div style="font-size:var(--fs-sm); color:var(--muted); margin-bottom:8px;">
|
||||
Planul Gratuit include {{ monthly_limit_val }} prestatii/luna prin dashboard-ul web.
|
||||
{% if plan_limita_atinsa|default(false) %}
|
||||
Limita lunara a fost atinsa — trimiterile noi sunt blocate pana la inceputul lunii urmatoare.
|
||||
{% elif plan_warn|default(false) %}
|
||||
Te apropii de limita lunara.
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<div style="font-size:var(--fs-sm); color:var(--muted); padding:8px 10px;
|
||||
border:1px solid var(--line); border-radius:6px; margin-top:4px;">
|
||||
Vrei sa treci pe Standard, Pro sau Premium?
|
||||
Contacteaza-ne pentru alocare manuala — nu exista inca plata self-service.
|
||||
<strong>Pro</strong> adauga import prin API; <strong>Standard</strong> si
|
||||
<strong>Premium</strong> ridica limita de volum.
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<!-- Sectiunea: Date firma (US-002) -->
|
||||
<div style="margin-bottom:20px; padding-bottom:20px; border-bottom:1px solid var(--line);">
|
||||
<h3 style="font-size:13px; color:var(--muted); font-weight:500; margin:0 0 8px; text-transform:uppercase; letter-spacing:.04em;">Date firma</h3>
|
||||
|
||||
{% if date_firma_mesaj %}
|
||||
<div class="flash" style="margin-bottom:12px;">{{ date_firma_mesaj }}</div>
|
||||
{% endif %}
|
||||
|
||||
{% if date_firma_eroare %}
|
||||
<div class="banner" style="margin-bottom:12px; padding:8px 12px;">{{ date_firma_eroare }}</div>
|
||||
{% endif %}
|
||||
|
||||
<form hx-post="/cont/date-firma"
|
||||
hx-target="#card-cont"
|
||||
hx-swap="outerHTML">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<p style="margin:0 0 8px;">
|
||||
<label style="font-size:13px; color:var(--muted);">Companie</label><br>
|
||||
<input type="text" name="companie" required
|
||||
value="{{ account_meta.name or '' }}"
|
||||
style="width:100%; max-width:340px;"
|
||||
placeholder="Numele firmei (ex. Service Auto SRL)">
|
||||
</p>
|
||||
<p style="margin:0 0 8px;">
|
||||
<label style="font-size:13px; color:var(--muted);">Email contact</label><br>
|
||||
<input type="email" name="email" required
|
||||
value="{{ account_meta.email or '' }}"
|
||||
style="width:100%; max-width:340px;"
|
||||
placeholder="contact@firma.ro">
|
||||
</p>
|
||||
<p style="margin:0 0 12px;">
|
||||
<label style="font-size:13px; color:var(--muted);">CUI (cod unic de identificare)</label><br>
|
||||
<input type="text" name="cui" required
|
||||
value="{{ account_meta.cui or '' }}"
|
||||
style="width:100%; max-width:340px;"
|
||||
placeholder="RO12345678">
|
||||
</p>
|
||||
<button type="submit">Salveaza datele firmei</button>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- Sectiunea: Cheia mea API -->
|
||||
<div style="margin-bottom:20px; padding-bottom:20px; border-bottom:1px solid var(--line);">
|
||||
<h3 style="font-size:13px; color:var(--muted); font-weight:500; margin:0 0 8px; text-transform:uppercase; letter-spacing:.04em;">Cheia mea API</h3>
|
||||
|
||||
74
app/web/templates/_editare_preview_modal.html
Normal file
74
app/web/templates/_editare_preview_modal.html
Normal file
@@ -0,0 +1,74 @@
|
||||
{# _editare_preview_modal.html — fragment de editare rand preview in modalul global.
|
||||
US-006 (PRD 5.12): swap-uit in #detaliu-modal-body de butonul Editeaza din preview.
|
||||
US-007 (PRD 5.12): butonul 'Confirma valorile' apare DOAR pe randurile needs_review
|
||||
(T2): trimite CSRF POST la /confirma-review, inchide modalul via HX-Trigger-After-Settle.
|
||||
|
||||
Necesita din context:
|
||||
import_id — id batch import
|
||||
row_index — index rand (0-based)
|
||||
csrf_token — token CSRF
|
||||
vin — VIN pentru titlu
|
||||
stare_css — clasa CSS pill (ex. "s-ok")
|
||||
stare_eticheta — text pill (ex. "Gata de trimis")
|
||||
message — mesaj de eroare general (None daca nu e)
|
||||
is_needs_review — True daca randul e in starea needs_review (afiseaza butonul Confirma)
|
||||
+ variabilele pentru _form_editare.html:
|
||||
form_nr, form_vin, form_data, form_odo_final, form_odo_initial
|
||||
err_map, fix_map, vin_context, btn_label
|
||||
#}
|
||||
<div class="card" style="border:none; padding:0; margin:0;">
|
||||
|
||||
{# Header cu heading accesibil (aria-labelledby al dialogului) #}
|
||||
<div style="display:flex; align-items:center; gap:8px; flex-wrap:wrap; margin:0 0 12px;">
|
||||
<h2 id="detaliu-modal-titlu" style="font-size:15px; margin:0;">
|
||||
Editare rand {{ row_index + 1 }}
|
||||
{% if vin %}<span class="muted" style="font-weight:400; font-size:13px;">· {{ vin }}</span>{% endif %}
|
||||
</h2>
|
||||
<span class="pill {{ stare_css }}" style="font-size:11px;">{{ stare_eticheta }}</span>
|
||||
</div>
|
||||
|
||||
{% if message %}
|
||||
<div class="flash" style="border-color:var(--err); background:color-mix(in srgb, var(--err) 12%, var(--card)); margin-bottom:12px;"
|
||||
role="alert">{{ message }}</div>
|
||||
{% endif %}
|
||||
|
||||
<form hx-post="/_import/{{ import_id }}/rand/{{ row_index }}/editeaza"
|
||||
hx-target="#detaliu-modal-body"
|
||||
hx-swap="innerHTML"
|
||||
hx-disabled-elt="find button"
|
||||
hx-on::response-error="this.querySelector && this.querySelector('.rand-eroare-banner') && (this.querySelector('.rand-eroare-banner').style.display='block');">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
|
||||
<div class="rand-eroare-banner" role="alert"
|
||||
style="display:none; margin-bottom:10px; padding:8px 12px; border:1px solid var(--err);
|
||||
background:color-mix(in srgb, var(--err) 12%, var(--card)); border-radius:6px; font-size:13px;">
|
||||
Salvarea nu a reusit (retea / sesiune). Valorile introduse sunt pastrate — reincearca.
|
||||
</div>
|
||||
|
||||
{# with_cancel=True: _form_editare.html randeaza Salveaza + Anuleaza pe acelasi
|
||||
rand (sistemul .act: desktop text, mobil iconite Lucide 44px alaturate). #}
|
||||
{% set with_cancel = true %}
|
||||
{% include "_form_editare.html" %}
|
||||
</form>
|
||||
|
||||
{% if is_needs_review %}
|
||||
{# T2 (US-007): Butonul 'Confirma valorile' apare DOAR pe randurile needs_review.
|
||||
POST separat (form propriu) la /confirma-review cu CSRF. Raspunsul inchide
|
||||
modalul via HX-Trigger-After-Settle: inchideModal + swap OOB randul si countorii. #}
|
||||
<form hx-post="/_import/{{ import_id }}/rand/{{ row_index }}/confirma-review"
|
||||
hx-target="#detaliu-modal-body"
|
||||
hx-swap="innerHTML"
|
||||
hx-disabled-elt="find button"
|
||||
style="margin-top:12px; border-top:1px solid var(--line); padding-top:12px;">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<p class="muted" style="font-size:13px; margin:0 0 8px;">
|
||||
Valorile sunt corecte si doriesti sa includi acest rand la trimitere la RAR?
|
||||
</p>
|
||||
<button type="submit"
|
||||
style="min-height:44px; padding:8px 18px;
|
||||
background:var(--ok, #2a7); color:#fff; border-color:transparent;">
|
||||
Confirma valorile
|
||||
</button>
|
||||
</form>
|
||||
{% endif %}
|
||||
</div>
|
||||
@@ -1,5 +1,5 @@
|
||||
{#
|
||||
_eroare.html — macro card_erori(erori) (US-006, PRD 5.4).
|
||||
_eroare.html — macro card_erori(erori).
|
||||
|
||||
Primeste o lista de dict-uri cu cheile: problema, cauza, fix, field (sau None).
|
||||
Afiseaza 3 niveluri intr-un bloc scannabil:
|
||||
|
||||
113
app/web/templates/_form_editare.html
Normal file
113
app/web/templates/_form_editare.html
Normal file
@@ -0,0 +1,113 @@
|
||||
{# _form_editare.html — partial partajat slim: campurile vehicul/data/odo + obs + chips prestatii.
|
||||
US-007 (PRD 5.15): redesign slim cu VIN unic, Observatii textarea, chips prestatii (E4),
|
||||
si reveal dinamic odometru initial cand chips contin R-ODO/I-ODO (D10c, E6 server-driven).
|
||||
|
||||
Inclus cu {% include "_form_editare.html" %} INSIDE un <form> element al
|
||||
template-ului parinte. Acel parinte pune form-ul, CSRF-ul si orice campuri
|
||||
suplimentare.
|
||||
|
||||
Variabile necesare din context (setate de parinte inainte de include):
|
||||
form_nr — valoare curenta nr_inmatriculare
|
||||
form_vin — valoare curenta vin
|
||||
form_data — valoare curenta data_prestatie (YYYY-MM-DD sau brut)
|
||||
form_odo_final — valoare curenta odometru_final
|
||||
form_odo_initial — valoare curenta odometru_initial
|
||||
obs_val — valoare curenta obs (Observatii), text liber (default '')
|
||||
prestatii_chips — list of {cod_prestatie, cod_op_service, denumire} (default [])
|
||||
nomenclator_rar — list of {cod_prestatie, nume_prestatie} pentru picker (default [])
|
||||
has_r_odo — True daca chips contin R-ODO/I-ODO (server-computed, default False)
|
||||
form_chips_url — URL pentru HTMX chip endpoint (default '/form-chips')
|
||||
err_map — dict {field_name: mesaj_eroare} (poate fi {})
|
||||
fix_map — dict {field_name: hint_fix} (poate fi {})
|
||||
vin_context — string VIN pentru aria-label (poate fi '')
|
||||
btn_label — eticheta butonului primar (ex. 'Salveaza si retrimite')
|
||||
#}
|
||||
{% from "_macros.html" import camp, icon %}
|
||||
|
||||
{# === 1. VIN — camp unic (fara "Confirma VIN"; contractul RAR cere un singur VIN) === #}
|
||||
{{ camp('vin', 'VIN (serie sasiu)', form_vin, slim=True, mono=True,
|
||||
err_map=err_map, fix_map=fix_map, vin_context=vin_context) }}
|
||||
|
||||
{# === 2. Data prestatie + Nr. inmatriculare — grila 2 coloane === #}
|
||||
<div style="display:grid; grid-template-columns:1fr 1fr; gap:0 12px;">
|
||||
{{ camp('data_prestatie', 'Data prestatiei', form_data, tip='date', slim=True,
|
||||
err_map=err_map, fix_map=fix_map, vin_context=vin_context) }}
|
||||
{{ camp('nr_inmatriculare', 'Numar inmatriculare', form_nr, slim=True, mono=True,
|
||||
err_map=err_map, fix_map=fix_map, vin_context=vin_context) }}
|
||||
</div>
|
||||
|
||||
{# === 3. Observatii (obs) — textarea liber, US-005 === #}
|
||||
<div class="camp-slim">
|
||||
<label for="c-obs">Observatii (operatiile efectuate)</label>
|
||||
<textarea id="c-obs" name="obs" rows="2"
|
||||
aria-label="Observatii (operatiile efectuate){% if vin_context %} (VIN: {{ vin_context }}){% endif %}"
|
||||
placeholder="ex: Revizie; schimbare placute frana">{{ obs_val or '' }}</textarea>
|
||||
</div>
|
||||
|
||||
{# === 4. Prestatii chips (E4 server-driven, US-007) === #}
|
||||
{% set form_chips_url = form_chips_url or '/form-chips' %}
|
||||
{% set chips_section_id = 'chips-section' %}
|
||||
{% include "_chips_prestatii.html" %}
|
||||
|
||||
{# === 5. Odometru final — intotdeauna vizibil === #}
|
||||
{{ camp('odometru_final', 'Odometru final (km)', form_odo_final, slim=True, mono=True,
|
||||
err_map=err_map, fix_map=fix_map, vin_context=vin_context) }}
|
||||
|
||||
{# === 6. Odometru initial — reveal dinamic server cand chips contin R-ODO/I-ODO (D10c) ===
|
||||
has_r_odo=True (server-computed din lista de chips): sectiune vizibila cu marcaj warn.
|
||||
has_r_odo=False: hint discret, campul optional si vizual neutru. #}
|
||||
{% if has_r_odo %}
|
||||
<div class="odo-initial-warn"
|
||||
style="border-left:2px solid var(--warn);padding-left:10px;margin-left:-2px;">
|
||||
<div class="camp-slim">
|
||||
<label for="c-odometru_initial" style="color:var(--warn);">
|
||||
Odometru initial (km) · necesar pentru R-ODO
|
||||
</label>
|
||||
<input id="c-odometru_initial" type="text" name="odometru_initial"
|
||||
value="{{ form_odo_initial or '' }}"
|
||||
class="camp-mono"
|
||||
required
|
||||
aria-required="true"
|
||||
style="border-color:color-mix(in srgb,var(--warn) 50%,var(--line));{% if err_map.get('odometru_initial') %}border-color:var(--err);{% endif %}"
|
||||
aria-label="Odometru initial (VIN: {{ vin_context or '' }}) — necesar pentru R-ODO"
|
||||
{% if err_map.get('odometru_initial') %}aria-invalid="true"{% endif %}>
|
||||
{% if err_map.get('odometru_initial') %}
|
||||
<div class="s-error" style="font-size:12px;margin-top:2px;">{{ err_map.get('odometru_initial') }}</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
{% else %}
|
||||
{# Hint discret cand nu e necesar #}
|
||||
<div class="camp-slim">
|
||||
<label for="c-odometru_initial" style="color:var(--muted);">Odometru initial (km)</label>
|
||||
<input id="c-odometru_initial" type="text" name="odometru_initial"
|
||||
value="{{ form_odo_initial or '' }}"
|
||||
class="camp-mono"
|
||||
style="{% if err_map.get('odometru_initial') %}border-color:var(--err);{% endif %}"
|
||||
aria-label="Odometru initial (optional){% if vin_context %} (VIN: {{ vin_context }}){% endif %}"
|
||||
{% if err_map.get('odometru_initial') %}aria-invalid="true"{% endif %}>
|
||||
{% if err_map.get('odometru_initial') %}
|
||||
<div class="s-error" style="font-size:12px;margin-top:2px;">{{ err_map.get('odometru_initial') }}</div>
|
||||
{% endif %}
|
||||
<span style="font-size:10px;color:var(--muted);font-style:italic;">
|
||||
Odometru initial se cere doar pentru coduri R-ODO / I-ODO.
|
||||
</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === 7. Buton primar parametrizat ===
|
||||
with_cancel=True (modal editare preview): Salveaza + Anuleaza pe ACELASI rand,
|
||||
sistemul .act (desktop = text alaturat; mobil = doua iconite Lucide 44px alaturate).
|
||||
Implicit (ex. _trimitere_detaliu): un singur buton text, neschimbat. #}
|
||||
{% if with_cancel %}
|
||||
<div class="act-group" style="margin-top:14px;">
|
||||
<button type="submit" class="act act-primary" aria-label="{{ btn_label or 'Salveaza' }}">
|
||||
<span class="act-tx">{{ btn_label or 'Salveaza' }}</span>{{ icon('save') }}</button>
|
||||
<button type="button" class="act" aria-label="{{ cancel_label or 'Renunta' }}" data-modal-close>
|
||||
<span class="act-tx">{{ cancel_label or 'Renunta' }}</span>{{ icon('x') }}</button>
|
||||
</div>
|
||||
{% else %}
|
||||
<div style="margin-top:14px;">
|
||||
<button type="submit">{{ btn_label or 'Salveaza' }}</button>
|
||||
</div>
|
||||
{% endif %}
|
||||
@@ -1,6 +1,6 @@
|
||||
{# _jurnal.html — tab Jurnal de aplicatie (US-006, PRD 5.6).
|
||||
{# _jurnal.html — tab Jurnal de aplicatie.
|
||||
Lista paginata de evenimente (app_events), redactate la scriere. Filtre tip/nivel/
|
||||
data + (admin) cont. Stil consistent cu tabelele PRD 5.5 (.tablewrap). #}
|
||||
data + (admin) cont. #}
|
||||
<section id="jurnal-section" aria-labelledby="jurnal-heading">
|
||||
<div class="card">
|
||||
<div style="display:flex; align-items:center; gap:8px; flex-wrap:wrap; margin:0 0 12px;">
|
||||
|
||||
@@ -1,27 +1,80 @@
|
||||
{# Macro-uri partajate intre template-urile de import si mapari. #}
|
||||
|
||||
{# US-003 (5.5): comutator COMPACT "In coada" — Manual <-> Auto pe un singur rand, fara
|
||||
proza inline (explicatia traieste o singura data in panoul Ajutor al cardului Mapari).
|
||||
Inlocuieste varianta verbose din 3.6 care injecta 3 randuri de text pe FIECARE linie de
|
||||
tabel (randuri inalte -> Salveaza/Sterge ieseau din viewport).
|
||||
{# US-002 (PRD 5.11): autosend_toggle neutralizat — auto_send nu mai tine randuri (US-001).
|
||||
Simbolul pastrat (apelat in _mapari.html, _preview_import.html, _trimitere_detaliu.html)
|
||||
dar intoarce string gol. Coloanele DB raman (default 1, ne-citite pentru hold). #}
|
||||
{% macro autosend_toggle(form_id='', checked=True, label='') -%}{%- endmacro %}
|
||||
|
||||
INVARIANT BACKEND (nealterat din 3.6): control = checkbox cu `name="auto_send" value="true"`
|
||||
si SEMANTICA DE PREZENTA (bifat -> trimite "true" -> True; nebifat -> camp absent -> False).
|
||||
E singura forma compatibila cu AMBELE parsere: `Form(bool)` la /mapari SI `bool(form.get())`
|
||||
la /_import/.../mapare-operatie. Radio Auto/Manual cu value="false" ar trimite campul prezent
|
||||
pe "Manual" -> `bool("false")` = True la import (regresie tacuta). De aceea comutator vizual
|
||||
Manual<->Auto peste checkbox, NU doua radio-uri. Zero atingere backend.
|
||||
- form_id: leaga input-ul de un <form> extern (necesar in celulele de tabel).
|
||||
- checked: starea STOCATA per mapare (H4) — bifat = Auto. #}
|
||||
{% macro autosend_toggle(form_id='', checked=True, label='') -%}
|
||||
<label class="autosend-toggle"
|
||||
title="Bifat = Auto: pune automat in coada la fisierele viitoare cu aceasta operatie. Nebifat = Manual: tine pentru verificare; nimic nu pleaca la RAR pana confirmi."
|
||||
style="display:inline-flex; align-items:center; justify-content:center; gap:8px; min-height:36px; cursor:pointer;">
|
||||
{%- if label %}<span class="muted" style="font-size:13px;">{{ label }}</span>{% endif %}
|
||||
<input type="checkbox" name="auto_send" value="true"
|
||||
{%- if form_id %} form="{{ form_id }}"{% endif %}
|
||||
{%- if checked %} checked{% endif %}
|
||||
aria-label="In coada automat (Auto) pentru aceasta operatie"
|
||||
style="width:18px; height:18px; cursor:pointer; accent-color:var(--accent);">
|
||||
</label>
|
||||
{# US-005 (PRD 5.12): macro `camp` partajat — extras din _trimitere_detaliu.html si
|
||||
_preview_rand.html. Suporta tip='date' (calendar nativ, D#10/R3) si tip='text' (default).
|
||||
|
||||
Parametri:
|
||||
nome — name="" al input-ului (si cheie in err_map/fix_map)
|
||||
eticheta — text pentru label
|
||||
valoare — valoarea curenta (pre-fill)
|
||||
tip — type="" al input-ului: 'text' (default) sau 'date' (calendar nativ)
|
||||
err_map — dict {field_name: mesaj_eroare}; default {}
|
||||
fix_map — dict {field_name: hint_fix}; default {}
|
||||
vin_context — string VIN pentru aria-label cu context (default '')
|
||||
id_prefix — prefix pentru id="" al input-ului (default 'c'; preview poate folosi 'e-N')
|
||||
#}
|
||||
{% macro camp(nome, eticheta, valoare, tip='text', err_map={}, fix_map={}, vin_context='', id_prefix='c', slim=False, mono=False) %}
|
||||
{# slim=False: randare clasica (neschimbata). slim=True: varianta compacta (.camp-slim) din US-002 PRD 5.15:
|
||||
label 11px muted deasupra, input ~30px, fundal --card2.
|
||||
mono=True (valid numai cu slim=True): adauga clasa 'camp-mono' pe input pentru campuri
|
||||
VIN/odometru/nr (IBM Plex Mono, prin .camp-slim .camp-mono din base.html). #}
|
||||
<div {% if slim %}class="camp-slim"{% else %}style="margin-bottom:10px;"{% endif %}>
|
||||
<label for="{{ id_prefix }}-{{ nome }}"{% if not slim %} class="muted" style="font-size:12px; display:block;"{% endif %}>{{ eticheta }}</label>
|
||||
{% if tip == 'date' %}
|
||||
{# D#10/R3: degradare grijulie pentru valori ne-YYYY-MM-DD.
|
||||
Daca valoarea nu e in formatul corect, inputul ramane gol + hint + hidden cu valoarea bruta
|
||||
(ca sa nu se piarda tacut la submit). #}
|
||||
{%- set _dp_ok = (valoare and valoare|length == 10 and valoare[4:5] == '-' and valoare[7:8] == '-') -%}
|
||||
<input id="{{ id_prefix }}-{{ nome }}" type="date" name="{{ nome }}"
|
||||
value="{{ valoare if _dp_ok else '' }}"
|
||||
{% if slim and mono %}class="camp-mono"{% endif %}
|
||||
style="{% if not slim %}width:100%; {% endif %}{% if err_map.get(nome) %}border-color:var(--err);{% endif %}"
|
||||
aria-label="{{ eticheta }}{% if vin_context %} (VIN: {{ vin_context }}){% endif %}"
|
||||
{% if err_map.get(nome) %}aria-invalid="true"{% endif %}>
|
||||
{% if not _dp_ok and valoare %}
|
||||
<input type="hidden" name="data_prestatie_raw" value="{{ valoare }}">
|
||||
<span class="camp-fix" style="font-size:12px;">Valoarea originala: {{ valoare }}</span>
|
||||
{% endif %}
|
||||
{% else %}
|
||||
<input id="{{ id_prefix }}-{{ nome }}" type="{{ tip }}" name="{{ nome }}"
|
||||
value="{{ valoare or '' }}"
|
||||
{% if slim and mono %}class="camp-mono"{% endif %}
|
||||
style="{% if not slim %}width:100%; {% endif %}{% if err_map.get(nome) %}border-color:var(--err);{% endif %}"
|
||||
{% if vin_context %}aria-label="{{ eticheta }} (VIN: {{ vin_context }})"{% endif %}
|
||||
{% if err_map.get(nome) %}aria-invalid="true"{% endif %}>
|
||||
{% endif %}
|
||||
{% if err_map.get(nome) %}
|
||||
<div class="s-error" style="font-size:12px; margin-top:2px;">{{ err_map.get(nome) }}</div>
|
||||
{% endif %}
|
||||
{% if fix_map.get(nome) %}
|
||||
<span class="camp-fix" style="font-size:12px;">{{ fix_map.get(nome) }}</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endmacro %}
|
||||
|
||||
{# PRD 5.13 — sistem butoane de actiune responsive.
|
||||
CSS-ul aferent (.act, .act-tx, .act-ic, .act-save, .act-del, .act-group)
|
||||
este definit in base.html.
|
||||
Desktop: se afiseaza textul (.act-tx); mobil: se afiseaza iconita (.act-ic). #}
|
||||
|
||||
{% macro icon(name) -%}
|
||||
<svg class="act-ic" viewBox="0 0 24 24" width="18" height="18" fill="none" stroke="currentColor"
|
||||
stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true">
|
||||
{%- if name == 'save' -%}<path d="M19 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11l5 5v11a2 2 0 0 1-2 2z"/><polyline points="17 21 17 13 7 13 7 21"/><polyline points="7 3 7 8 15 8"/>
|
||||
{%- elif name == 'trash' -%}<polyline points="3 6 5 6 21 6"/><path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6m3 0V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2"/><line x1="10" y1="11" x2="10" y2="17"/><line x1="14" y1="11" x2="14" y2="17"/>
|
||||
{%- elif name == 'edit' -%}<path d="M11 4H4a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7"/><path d="M18.5 2.5a2.121 2.121 0 0 1 3 3L12 15l-4 1 1-4 9.5-9.5z"/>
|
||||
{%- elif name == 'plus' -%}<line x1="12" y1="5" x2="12" y2="19"/><line x1="5" y1="12" x2="19" y2="12"/>
|
||||
{%- elif name == 'x' -%}<line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/>
|
||||
{%- endif -%}
|
||||
</svg>
|
||||
{%- endmacro %}
|
||||
|
||||
{% macro act_btn(label, ic, kind='', attrs='') -%}
|
||||
<button class="act{% if kind %} act-{{ kind }}{% endif %}" aria-label="{{ label }}" {{ attrs | safe }}>
|
||||
<span class="act-tx">{{ label }}</span>{{ icon(ic) }}</button>
|
||||
{%- endmacro %}
|
||||
|
||||
@@ -4,7 +4,7 @@
|
||||
/* Selectul de cod RAR e principalul vinovat de latimea tabelelor de mapari. Il limitam ca
|
||||
tabelul sa incapa in card fara scroll orizontal -> coloana Actiuni (kebab) ramane vizibila. */
|
||||
#mapari-section td select { width:100%; max-width:240px; min-width:150px; }
|
||||
/* US-007 (R12): in card per rand (sub 767px) selectul/inputurile umplu cardul. */
|
||||
/* In card per rand (sub 767px) selectul/inputurile umplu cardul. */
|
||||
@media (max-width:767px) {
|
||||
#mapari-section td select, #mapari-section td input[type=text] { max-width:none; min-width:0; }
|
||||
}
|
||||
@@ -18,29 +18,9 @@
|
||||
<!-- Sectiunea 1: De rezolvat (operatii needs_mapping) -->
|
||||
<!-- ============================================================ -->
|
||||
<div class="card">
|
||||
{# US-005 (5.5): antet standard + link Ajutor ca <details> nativ (fara JS). Toata proza
|
||||
care inainte se repeta inline (scopul maparilor, Auto/Manual) traieste acum AICI,
|
||||
o singura data, ascunsa implicit. #}
|
||||
<h2 style="font-size:15px; margin:0 0 8px;">De rezolvat</h2>
|
||||
<details class="ajutor-mapari" style="margin:0 0 12px;">
|
||||
<summary class="cardlink" style="display:inline-flex; color:var(--accent); cursor:pointer; padding:4px 0;">Ajutor</summary>
|
||||
<div class="muted" style="font-size:13px; margin-top:8px; max-width:680px;">
|
||||
Maparile leaga o operatie din softul tau (cod intern ROAAUTO) de un cod RAR oficial.
|
||||
Operatiile necunoscute raman blocate in <span class="s-needs_mapping">needs_mapping</span>
|
||||
si NU pleaca la RAR pana le mapezi. Sugestiile (%) vin din potrivire fuzzy pe denumire —
|
||||
verifica-le inainte sa salvezi. <strong>In coada</strong>: <strong>Auto</strong> = la
|
||||
urmatoarele fisiere cu aceasta operatie randurile intra automat in coada;
|
||||
<strong>Manual</strong> = raman pentru verificare, nimic nu pleaca la RAR pana confirmi.
|
||||
La schimbarea unui cod, submission-urile blocate pe acea operatie se re-rezolva automat.
|
||||
</div>
|
||||
</details>
|
||||
|
||||
{% if not pending %}
|
||||
<div class="empty">
|
||||
Nicio operatie nemapata — tot ce a venit s-a tradus in coduri RAR.
|
||||
<a href="/?tab=acasa">Importa un fisier nou</a> daca vrei sa adaugi prezentari.
|
||||
</div>
|
||||
{% else %}
|
||||
{% if pending %}
|
||||
<div data-dt="10">
|
||||
<div class="dt-tools">
|
||||
<input type="search" data-dt-search class="dt-search"
|
||||
@@ -52,13 +32,13 @@
|
||||
<th>Operatie</th>
|
||||
<th>Sugestii</th>
|
||||
<th>Cod RAR</th>
|
||||
<th>In coada</th>
|
||||
<th></th>
|
||||
</tr></thead>
|
||||
<tbody>
|
||||
{% for e in pending %}
|
||||
{% set top = e.suggestions[0] if e.suggestions else None %}
|
||||
{% set preselect = top.cod_prestatie if (top and top.score >= 60) else '' %}
|
||||
{# L14-S6: pre-selectare din sugestie_principala (GOLD/SILVER/embedding) > fuzzy #}
|
||||
{% set preselect = e.sugestie_principala.cod_prestatie if e.sugestie_principala else (top.cod_prestatie if (top and top.score >= 60) else '') %}
|
||||
{# data-dt-row = haystack de cautare (randul contine un <select> cu tot nomenclatorul). #}
|
||||
<tr data-dt-row="{{ e.cod_op_service }} {{ e.denumire or '' }}
|
||||
{%- for s in e.suggestions[:3] %} {{ s.cod_prestatie }}{% endfor %}">
|
||||
@@ -66,17 +46,30 @@
|
||||
<form id="map-rez-{{ loop.index }}" hx-post="/mapari" hx-target="#mapari-section" hx-swap="outerHTML">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="hidden" name="cod_op_service" value="{{ e.cod_op_service }}">
|
||||
{# L14-S6: denumire pt record_human_validation in GOLD partajat #}
|
||||
<input type="hidden" name="denumire" value="{{ e.denumire or '' }}">
|
||||
</form>
|
||||
<div><strong>{{ e.cod_op_service }}</strong>
|
||||
<span class="pill" title="submission-uri blocate">{{ e.blocked }} blocate</span></div>
|
||||
<div class="muted">{{ e.denumire or '(fara denumire)' }}</div>
|
||||
</td>
|
||||
<td class="muted" style="font-size:12px;" data-eticheta="Sugestii">
|
||||
{# 5.18 US-007: badge sursa pe sugestia sistemului — confirmat (GOLD) / similar
|
||||
(SILVER+embedding k-NN) / non-operatie (pre-filtru NUL). Suggestion-only. #}
|
||||
{% if e.sugestie_principala %}
|
||||
{% if e.sugestie_principala.sursa == 'gold_partajat' %}
|
||||
<span class="sugg-sursa sugg-sursa--confirmat" title="cod confirmat de un operator">confirmat</span>
|
||||
{% else %}
|
||||
<span class="sugg-sursa sugg-sursa--similar" title="operatie similara deja vazuta (k-NN/exact)">similar</span>
|
||||
{% endif %}
|
||||
{% elif e.surse_sugestie and e.surse_sugestie.nul %}
|
||||
<span class="sugg-sursa sugg-sursa--nul" title="pare non-operatie (ITP/plata/discount...)">non-operatie</span>
|
||||
{% endif %}
|
||||
{% if e.suggestions %}
|
||||
{% for s in e.suggestions[:3] %}
|
||||
<span class="sugg">{{ s.cod_prestatie }} ({{ s.score|round|int }}%)</span>{% if not loop.last %}, {% endif %}
|
||||
{% endfor %}
|
||||
{% else %}—{% endif %}
|
||||
{% elif not e.sugestie_principala and not (e.surse_sugestie and e.surse_sugestie.nul) %}—{% endif %}
|
||||
</td>
|
||||
<td data-eticheta="Cod RAR">
|
||||
<select name="cod_prestatie" form="map-rez-{{ loop.index }}" required
|
||||
@@ -89,9 +82,6 @@
|
||||
{% endfor %}
|
||||
</select>
|
||||
</td>
|
||||
<td data-eticheta="In coada">
|
||||
{{ ui.autosend_toggle(form_id="map-rez-" ~ loop.index, checked=True) }}
|
||||
</td>
|
||||
<td>
|
||||
<button type="submit" form="map-rez-{{ loop.index }}">Salveaza</button>
|
||||
</td>
|
||||
@@ -117,8 +107,6 @@
|
||||
Nicio mapare salvata inca. Pe masura ce mapezi operatii, ele apar aici si le poti edita oricand.
|
||||
</div>
|
||||
{% else %}
|
||||
{# US-005 (5.5): proza explicativa mutata in panoul Ajutor de la "De rezolvat" (o singura data). #}
|
||||
|
||||
<div data-dt="10">
|
||||
<div class="dt-tools">
|
||||
<input type="search" data-dt-search class="dt-search"
|
||||
@@ -129,7 +117,6 @@
|
||||
<thead><tr>
|
||||
<th>Operatie</th>
|
||||
<th>Cod RAR</th>
|
||||
<th>In coada</th>
|
||||
<th>Actiuni</th>
|
||||
</tr></thead>
|
||||
<tbody>
|
||||
@@ -147,7 +134,7 @@
|
||||
<input type="hidden" name="cod_op_service" value="{{ m.cod_op_service }}">
|
||||
</form>
|
||||
<div><strong>{{ m.cod_op_service }}</strong></div>
|
||||
<div class="muted" style="font-size:12px;">
|
||||
<div class="muted map-acum" style="font-size:12px;">
|
||||
acum: {{ m.cod_prestatie }}{% if m.nume_prestatie %} — {{ m.nume_prestatie }}{% endif %}
|
||||
</div>
|
||||
</td>
|
||||
@@ -161,19 +148,12 @@
|
||||
{% endfor %}
|
||||
</select>
|
||||
</td>
|
||||
<td data-eticheta="In coada">
|
||||
{{ ui.autosend_toggle(form_id="map-salv-" ~ loop.index, checked=m.auto_send) }}
|
||||
</td>
|
||||
<td style="text-align:right; white-space:nowrap;" data-eticheta="Actiuni">
|
||||
{# Salveaza/Sterge in meniu contextual (kebab) — randul ramane ingust. Butoanele se
|
||||
leaga prin form= de cele doua form-uri hx-post definite in prima celula a randului. #}
|
||||
<details class="kebab">
|
||||
<summary aria-label="Actiuni pentru {{ m.cod_op_service }}">⋯</summary>
|
||||
<div class="kebab-menu">
|
||||
<button type="submit" form="map-salv-{{ loop.index }}">Salveaza</button>
|
||||
<button type="submit" form="map-del-{{ loop.index }}" class="danger">Sterge</button>
|
||||
</div>
|
||||
</details>
|
||||
{# Butoane act_btn (desktop: text; mobil: iconita 44px).
|
||||
data-dirty-form e citit de JS din base.html: la schimbarea select-ului din acelasi rand,
|
||||
JS adauga clasa "dirty" pe butonul de salvare (fundal --accent = modificari nesalvate). #}
|
||||
{{ ui.act_btn('Salveaza', 'save', 'save', 'type="submit" form="map-salv-' ~ loop.index ~ '" data-dirty-form="map-salv-' ~ loop.index ~ '"') }}
|
||||
{{ ui.act_btn('Sterge', 'trash', 'del', 'type="submit" form="map-del-' ~ loop.index ~ '"') }}
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
@@ -187,7 +167,91 @@
|
||||
</div>
|
||||
|
||||
<!-- ============================================================ -->
|
||||
<!-- Sectiunea 3: Formate de coloane salvate (column_mappings) -->
|
||||
<!-- Sectiunea 3: Reguli automate pe text (operation_text_rules) -->
|
||||
<!-- ============================================================ -->
|
||||
<div class="card">
|
||||
<h2 style="font-size:15px; margin:0 0 8px;">Reguli automate (text)</h2>
|
||||
<p class="muted" style="margin:0 0 12px; font-size:13px; max-width:680px;">
|
||||
O regula leaga orice operatie al carei text <strong>contine</strong> (nu egal, ci substring)
|
||||
un cuvant de un cod RAR. Util pentru operatii fara cod intern: ex. orice operatie care
|
||||
<em>contine</em> „verificare" primeste codul ales. Match insensibil la majuscule/diacritice.
|
||||
</p>
|
||||
|
||||
{% if not text_rules %}
|
||||
<div class="empty" style="margin-bottom:12px;">
|
||||
Inca nu ai reguli. Ex: operatia contine «verificare» → OE-2.
|
||||
Mapeaza automat operatii similare fara cod intern. Adauga prima regula mai jos.
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<div class="tablewrap tabel-card">
|
||||
<table>
|
||||
<thead><tr>
|
||||
<th>Daca operatia contine</th>
|
||||
<th>Cod RAR</th>
|
||||
<th>Actiuni</th>
|
||||
</tr></thead>
|
||||
<tbody>
|
||||
{% for r in text_rules %}
|
||||
<tr>
|
||||
<td data-eticheta="Daca operatia contine">
|
||||
<form id="rt-del-{{ loop.index }}" hx-post="/mapari/reguli-text/sterge"
|
||||
hx-target="#mapari-section" hx-swap="outerHTML"
|
||||
hx-confirm="Stergi regula «{{ r.pattern }}»?">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="hidden" name="pattern" value="{{ r.pattern }}">
|
||||
</form>
|
||||
<div>contine <strong>«{{ r.pattern }}»</strong></div>
|
||||
</td>
|
||||
<td class="muted" style="font-size:12px;" data-eticheta="Cod RAR">
|
||||
{{ r.cod_prestatie }}
|
||||
</td>
|
||||
<td style="text-align:right; white-space:nowrap;">
|
||||
{{ ui.act_btn('Sterge', 'trash', 'del', 'type="submit" form="rt-del-' ~ loop.index ~ '"') }}
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
{# Rand de adaugare (mereu prezent ca placeholder, inclusiv in empty state). #}
|
||||
<tr>
|
||||
<td data-eticheta="Daca operatia contine">
|
||||
<form id="rt-add" hx-post="/mapari/reguli-text" hx-target="#mapari-section" hx-swap="outerHTML">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="text" name="pattern" required
|
||||
placeholder="ex. verificare"
|
||||
aria-label="Text continut in operatie"
|
||||
style="width:100%; max-width:240px;"
|
||||
hx-post="/mapari/reguli-text/preview"
|
||||
hx-trigger="keyup delay:400ms"
|
||||
hx-target="#rt-preview"
|
||||
hx-swap="innerHTML"
|
||||
hx-include="#rt-add">
|
||||
</form>
|
||||
</td>
|
||||
<td data-eticheta="Cod RAR">
|
||||
<select name="cod_prestatie" form="rt-add" required aria-label="Cod RAR pentru regula text">
|
||||
<option value="">— alege cod RAR —</option>
|
||||
{% for n in nomenclator %}
|
||||
<option value="{{ n.cod_prestatie }}">{{ n.cod_prestatie }} — {{ n.nume_prestatie }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</td>
|
||||
<td style="text-align:right; white-space:nowrap;">
|
||||
<button type="submit" form="rt-add">Adauga</button>
|
||||
</td>
|
||||
</tr>
|
||||
{# Preview pre-salvare: cate operatii nemapate potriveste pattern-ul. #}
|
||||
<tr>
|
||||
<td colspan="3" style="padding-top:0;">
|
||||
<div id="rt-preview" aria-live="polite"></div>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ============================================================ -->
|
||||
<!-- Sectiunea 4: Formate de coloane salvate (column_mappings) -->
|
||||
<!-- ============================================================ -->
|
||||
<div class="card">
|
||||
<h2 style="font-size:15px; margin:0 0 12px;">Formate de coloane salvate</h2>
|
||||
@@ -242,9 +306,7 @@
|
||||
hx-confirm="Stergi acest format de coloane?">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="hidden" name="format_id" value="{{ f.id }}">
|
||||
<button type="submit" style="background:var(--card); color:var(--err); border-color:var(--err);">
|
||||
Sterge
|
||||
</button>
|
||||
{{ ui.act_btn('Sterge', 'trash', 'del', 'type="submit"') }}
|
||||
</form>
|
||||
</td>
|
||||
</tr>
|
||||
@@ -258,100 +320,4 @@
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
<!-- ============================================================ -->
|
||||
<!-- Sectiunea 4: Reguli automate pe text (operation_text_rules) -->
|
||||
<!-- ============================================================ -->
|
||||
<div class="card">
|
||||
<h2 style="font-size:15px; margin:0 0 8px;">Reguli automate (text)</h2>
|
||||
<p class="muted" style="margin:0 0 12px; font-size:13px; max-width:680px;">
|
||||
O regula leaga orice operatie al carei text <strong>contine</strong> (nu egal, ci substring)
|
||||
un cuvant de un cod RAR. Util pentru operatii fara cod intern: ex. orice operatie care
|
||||
<em>contine</em> „verificare" primeste codul ales. Match insensibil la majuscule/diacritice.
|
||||
<strong>In coada</strong>: implicit oprit — regula rezolva codul dar tine randul pentru
|
||||
verificare umana pana activezi „In coada".
|
||||
</p>
|
||||
|
||||
{% if not text_rules %}
|
||||
<div class="empty" style="margin-bottom:12px;">
|
||||
Inca nu ai reguli. Ex: operatia contine «verificare» → OE-2.
|
||||
Mapeaza automat operatii similare fara cod intern. Adauga prima regula mai jos.
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<div class="tablewrap tabel-card">
|
||||
<table>
|
||||
<thead><tr>
|
||||
<th>Daca operatia contine</th>
|
||||
<th>Cod RAR</th>
|
||||
<th>In coada</th>
|
||||
<th>Actiuni</th>
|
||||
</tr></thead>
|
||||
<tbody>
|
||||
{% for r in text_rules %}
|
||||
<tr>
|
||||
<td data-eticheta="Daca operatia contine">
|
||||
<form id="rt-del-{{ loop.index }}" hx-post="/mapari/reguli-text/sterge"
|
||||
hx-target="#mapari-section" hx-swap="outerHTML"
|
||||
hx-confirm="Stergi regula «{{ r.pattern }}»?">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="hidden" name="pattern" value="{{ r.pattern }}">
|
||||
</form>
|
||||
<div>contine <strong>«{{ r.pattern }}»</strong></div>
|
||||
</td>
|
||||
<td class="muted" style="font-size:12px;" data-eticheta="Cod RAR">
|
||||
{{ r.cod_prestatie }}
|
||||
</td>
|
||||
<td class="muted" style="font-size:12px;" data-eticheta="In coada">
|
||||
{% if r.auto_send %}Auto (in coada){% else %}Manual (verificare){% endif %}
|
||||
</td>
|
||||
<td style="text-align:right; white-space:nowrap;">
|
||||
<button type="submit" form="rt-del-{{ loop.index }}"
|
||||
style="background:var(--card); color:var(--err); border-color:var(--err);">
|
||||
Sterge
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
{# Rand de adaugare (mereu prezent ca placeholder, inclusiv in empty state). #}
|
||||
<tr>
|
||||
<td data-eticheta="Daca operatia contine">
|
||||
<form id="rt-add" hx-post="/mapari/reguli-text" hx-target="#mapari-section" hx-swap="outerHTML">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="text" name="pattern" required
|
||||
placeholder="ex. verificare"
|
||||
aria-label="Text continut in operatie"
|
||||
style="width:100%; max-width:240px;"
|
||||
hx-post="/mapari/reguli-text/preview"
|
||||
hx-trigger="keyup delay:400ms"
|
||||
hx-target="#rt-preview"
|
||||
hx-swap="innerHTML"
|
||||
hx-include="#rt-add">
|
||||
</form>
|
||||
</td>
|
||||
<td data-eticheta="Cod RAR">
|
||||
<select name="cod_prestatie" form="rt-add" required aria-label="Cod RAR pentru regula text">
|
||||
<option value="">— alege cod RAR —</option>
|
||||
{% for n in nomenclator %}
|
||||
<option value="{{ n.cod_prestatie }}">{{ n.cod_prestatie }} — {{ n.nume_prestatie }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</td>
|
||||
<td data-eticheta="In coada">
|
||||
{{ ui.autosend_toggle(form_id="rt-add", checked=False) }}
|
||||
</td>
|
||||
<td style="text-align:right; white-space:nowrap;">
|
||||
<button type="submit" form="rt-add">Adauga</button>
|
||||
</td>
|
||||
</tr>
|
||||
{# Preview pre-salvare (US-009): cate operatii nemapate potriveste pattern-ul. #}
|
||||
<tr>
|
||||
<td colspan="4" style="padding-top:0;">
|
||||
<div id="rt-preview" aria-live="polite"></div>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
@@ -1,8 +1,10 @@
|
||||
<div id="import-section">
|
||||
{% set pas = 2 %}{% include '_stepper.html' %}
|
||||
{% from '_eroare.html' import card_erori %}
|
||||
{# prima_inregistrare poate veni din context (web_upload_import) sau derivat din sample_rows #}
|
||||
{%- set prima_inreg = prima_inregistrare if prima_inregistrare is defined else (sample_rows[0] if sample_rows else none) -%}
|
||||
<div class="card">
|
||||
<h2 style="font-size:15px; margin:0 0 12px;">
|
||||
<h2 style="font-size:var(--fs-md); margin:0 0 12px;">
|
||||
Mapare coloane —
|
||||
<span class="muted" style="font-weight:400;">{{ filename or ("import #" ~ import_id) }}</span>
|
||||
</h2>
|
||||
@@ -18,18 +20,56 @@
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<p class="muted" style="margin:0 0 12px; font-size:13px;">
|
||||
<p class="muted" style="margin:0 0 12px; font-size:var(--fs-sm);">
|
||||
Asociaza fiecare coloana din fisier cu campul canonic corespunzator.
|
||||
Maparea se retine automat pentru fisiere cu acelasi antet.
|
||||
</p>
|
||||
|
||||
{# Tabel orizontal preview: antet + prima inregistrare (compatibilitate teste) #}
|
||||
<div class="tablewrap" style="margin-bottom:16px;">
|
||||
<table class="preview-antet" style="border-collapse:collapse; font-size:var(--fs-xs); width:100%; min-width:max-content;">
|
||||
<thead>
|
||||
<tr>
|
||||
{% for col in columns %}
|
||||
<th style="padding:4px 10px; text-align:left; background:var(--card); border:1px solid var(--line);
|
||||
white-space:nowrap; font-weight:600; font-size:var(--fs-xs); color:var(--ink);">
|
||||
{{ col }}
|
||||
</th>
|
||||
{% endfor %}
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{% if prima_inreg %}
|
||||
<tr>
|
||||
{% for col in columns %}
|
||||
{%- set val = prima_inreg.get(col, '') | string -%}
|
||||
<td style="padding:4px 10px; border:1px solid var(--line); white-space:nowrap;
|
||||
font-size:var(--fs-xs); color:var(--muted); max-width:160px; overflow:hidden; text-overflow:ellipsis;"
|
||||
title="{{ val }}">
|
||||
{{ val[:40] }}{% if val | length > 40 %}…{% endif %}
|
||||
</td>
|
||||
{% endfor %}
|
||||
</tr>
|
||||
{% else %}
|
||||
<tr>
|
||||
<td colspan="{{ columns | length }}"
|
||||
style="padding:6px 10px; border:1px solid var(--line); font-size:var(--fs-xs);
|
||||
color:var(--muted); font-style:italic; text-align:center;">
|
||||
Antet fara randuri de date
|
||||
</td>
|
||||
</tr>
|
||||
{% endif %}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<form hx-post="/_import/{{ import_id }}/mapare-coloane"
|
||||
hx-target="#import-section"
|
||||
hx-swap="outerHTML">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
|
||||
<div style="margin-bottom:8px; display:flex; align-items:center; gap:10px; flex-wrap:wrap;">
|
||||
<label for="format-data" style="font-size:13px; color:var(--muted);">
|
||||
<label for="format-data" style="font-size:var(--fs-sm); color:var(--muted);">
|
||||
Format data
|
||||
</label>
|
||||
<input type="text" id="format-data" name="format_data"
|
||||
@@ -37,67 +77,105 @@
|
||||
placeholder="ex: DD.MM.YYYY"
|
||||
style="max-width:160px;"
|
||||
aria-describedby="format-data-hint">
|
||||
<span id="format-data-hint" class="muted" style="font-size:12px;">
|
||||
<span id="format-data-hint" class="muted" style="font-size:var(--fs-xs);">
|
||||
sau YYYY-MM-DD, MM/DD/YYYY etc.
|
||||
</span>
|
||||
</div>
|
||||
|
||||
{% for col in columns %}
|
||||
{%- set sugg = fuzzy_suggestions.get(col, []) -%}
|
||||
{%- set best = sugg[0].camp_canonic if sugg else '' -%}
|
||||
<input type="hidden" name="colname" value="{{ col }}">
|
||||
<div class="maprow">
|
||||
<div class="mapcol grow">
|
||||
<div><strong>{{ col }}</strong></div>
|
||||
{% if sugg %}
|
||||
<div class="muted" style="font-size:12px; margin-top:2px;">
|
||||
sugestie: <span class="sugg">{{ sugg[0].camp_canonic }}
|
||||
({{ sugg[0].score | round | int }}%)</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
{%- set ns = namespace(samples=[]) -%}
|
||||
{%- for row in sample_rows -%}
|
||||
{%- if row.get(col) is not none and row.get(col) != '' -%}
|
||||
{%- set ns.samples = ns.samples + [row[col] | string] -%}
|
||||
{%- endif -%}
|
||||
{%- endfor -%}
|
||||
{% if ns.samples %}
|
||||
<div class="muted" style="font-size:11px; margin-top:2px;">
|
||||
ex: {{ ns.samples[:2] | join(", ") }}
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
<div class="mapcol" style="min-width:200px;">
|
||||
<label for="canon-{{ loop.index }}"
|
||||
style="display:block; font-size:12px; color:var(--muted); margin-bottom:2px;">
|
||||
Camp canonic
|
||||
</label>
|
||||
<select id="canon-{{ loop.index }}" name="canon">
|
||||
<option value="">— ignorat —</option>
|
||||
{% for field_key, field_label in canonical_fields %}
|
||||
<option value="{{ field_key }}"
|
||||
{% if field_key == best %}selected{% endif %}>
|
||||
{{ field_key }} — {{ field_label }}
|
||||
</option>
|
||||
{# Tabel mapare: coloana din fisier | exemplu | camp RAR (mockup 5.16 / US-013) #}
|
||||
<div class="tablewrap" style="margin-bottom:16px;">
|
||||
<table style="border-collapse:collapse; width:100%;">
|
||||
<thead>
|
||||
<tr>
|
||||
<th style="font-size:var(--fs-xs); width:34%; padding:6px 10px; text-align:left;
|
||||
background:var(--card2); border-bottom:1px solid var(--line); color:var(--muted);
|
||||
font-weight:600; text-transform:uppercase; letter-spacing:.04em;">
|
||||
Coloana din fisier
|
||||
</th>
|
||||
<th style="font-size:var(--fs-xs); width:28%; padding:6px 10px; text-align:left;
|
||||
background:var(--card2); border-bottom:1px solid var(--line); color:var(--muted);
|
||||
font-weight:600; text-transform:uppercase; letter-spacing:.04em;">
|
||||
Exemplu
|
||||
</th>
|
||||
<th style="font-size:var(--fs-xs); padding:6px 10px; text-align:left;
|
||||
background:var(--card2); border-bottom:1px solid var(--line); color:var(--muted);
|
||||
font-weight:600; text-transform:uppercase; letter-spacing:.04em;">
|
||||
Camp RAR
|
||||
</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{% for col in columns %}
|
||||
{%- set sugg = fuzzy_suggestions.get(col, []) -%}
|
||||
{%- set best = sugg[0].camp_canonic if sugg else '' -%}
|
||||
{%- set ns = namespace(samples=[]) -%}
|
||||
{%- for row in sample_rows -%}
|
||||
{%- if row.get(col) is not none and row.get(col) != '' -%}
|
||||
{%- set ns.samples = ns.samples + [row[col] | string] -%}
|
||||
{%- endif -%}
|
||||
{%- endfor -%}
|
||||
<tr style="border-bottom:1px solid var(--line);">
|
||||
<td style="padding:9px 10px; vertical-align:top;">
|
||||
<input type="hidden" name="colname" value="{{ col }}">
|
||||
<strong style="font-family:var(--font-mono); font-size:var(--fs-sm);">{{ col }}</strong>
|
||||
{% if sugg %}
|
||||
<div class="muted" style="font-size:var(--fs-xs); margin-top:3px;">
|
||||
sugestie: <span class="sugg">{{ sugg[0].camp_canonic }}
|
||||
({{ sugg[0].score | round | int }}%)</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td style="padding:9px 10px; vertical-align:top;">
|
||||
{% if ns.samples %}
|
||||
<span style="font-family:var(--font-mono); font-size:var(--fs-xs); color:var(--muted);">
|
||||
{{ ns.samples[:2] | join(", ") }}
|
||||
</span>
|
||||
{% else %}
|
||||
<span class="muted" style="font-size:var(--fs-xs);">—</span>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td style="padding:9px 10px; vertical-align:top;">
|
||||
<label for="canon-{{ loop.index }}"
|
||||
style="display:block; font-size:var(--fs-xs); color:var(--muted); margin-bottom:3px;">
|
||||
Camp canonic
|
||||
</label>
|
||||
<select id="canon-{{ loop.index }}" name="canon"
|
||||
style="width:100%; font-size:var(--fs-base); min-height:38px;">
|
||||
<option value="">— ignorat —</option>
|
||||
{% for field_key, field_label in canonical_fields %}
|
||||
<option value="{{ field_key }}"
|
||||
{% if field_key == best %}selected{% endif %}>
|
||||
{{ field_key }} — {{ field_label }}
|
||||
</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</td>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</div>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
{% endfor %}
|
||||
|
||||
<div style="margin-top:16px; display:flex; align-items:center; gap:12px; flex-wrap:wrap;">
|
||||
<button type="submit"
|
||||
style="min-height:44px; padding:10px 24px; font-size:14px;">
|
||||
{% if not prima_inreg %}disabled aria-disabled="true"{% endif %}
|
||||
style="min-height:44px; padding:10px 24px; font-size:var(--fs-md);{% if not prima_inreg %} opacity:0.5; cursor:not-allowed;{% endif %}">
|
||||
Salveaza si continua la preview
|
||||
</button>
|
||||
<span class="muted" style="font-size:12px;">
|
||||
{% if not prima_inreg %}
|
||||
<span style="font-size:var(--fs-xs); color:var(--err);">
|
||||
Fisierul nu contine randuri de date — incarca un fisier cu cel putin o inregistrare.
|
||||
</span>
|
||||
{% else %}
|
||||
<span class="muted" style="font-size:var(--fs-xs);">
|
||||
maparea se retine pentru fisiere cu acelasi antet
|
||||
</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<div style="margin-top:12px;">
|
||||
<a href="/" class="muted" style="font-size:13px;">Incarca alt fisier</a>
|
||||
<a href="/" class="muted" style="font-size:var(--fs-sm);">Incarca alt fisier</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,7 +1,5 @@
|
||||
{# US-002 (5.5): aceeasi grila standard ca tabelul Trimiteri (_submissions.html):
|
||||
.tablewrap > table, antet th standard (mostenit din base.html), cod in .pill,
|
||||
denumire ca text normal (singura coloana care se poate rupe pe randuri inguste),
|
||||
empty-state in .empty. Zero stiluri inline noi — totul vine din base.html. #}
|
||||
{# Aceeasi grila standard ca tabelul Trimiteri: cod in .pill, denumire ca text normal
|
||||
(singura coloana care se poate rupe pe randuri inguste), empty-state in .empty. #}
|
||||
{% if rows %}
|
||||
<div class="tablewrap">
|
||||
<table>
|
||||
|
||||
15
app/web/templates/_pills.html
Normal file
15
app/web/templates/_pills.html
Normal file
@@ -0,0 +1,15 @@
|
||||
{# Pill-uri de filtrare a starii, randate in bara de filtre (_coada.html) si re-randate
|
||||
prin OOB la fiecare reincarcare a tabelului (_submissions.html). Stare activa =
|
||||
status_filtru. "Toate" reseteaza filtrul; categoriile apar doar cand au n>0. #}
|
||||
<button type="button" class="pill-cat pill-cat-reset" data-status=""
|
||||
aria-pressed="{{ 'true' if not status_filtru else 'false' }}"
|
||||
onclick="filtreazaStare(this, '')">Toate</button>
|
||||
{% for pill in pills_categorii %}
|
||||
<button type="button" class="pill-cat" data-status="{{ pill.status }}"
|
||||
aria-pressed="{{ 'true' if status_filtru == pill.status else 'false' }}"
|
||||
style="color:var({{ pill.color_var }}); border-color:var({{ pill.color_var }});"
|
||||
onclick="filtreazaStare(this, '{{ pill.status }}')">
|
||||
{{ pill.label }}
|
||||
<span class="pill-cat-n" style="background:var({{ pill.color_var }});">{{ pill.n }}</span>
|
||||
</button>
|
||||
{% endfor %}
|
||||
@@ -1,13 +1,20 @@
|
||||
{% import '_macros.html' as ui %}
|
||||
<div id="import-section">
|
||||
{# reincarcaPreview (emis de /editeaza si /confirma-review prin HX-Trigger): preview-ul
|
||||
se reincarca COMPLET (rand + contoare + colaps deja-trimise corecte) in loc de OOB swap
|
||||
pe <tr> (fragil in htmx 1.9). Evidentierea + toast-ul randului salvat: base.html. #}
|
||||
<div id="import-section"
|
||||
hx-get="/_import/{{ import_id }}/preview"
|
||||
hx-trigger="reincarcaPreview from:body"
|
||||
hx-target="#import-section"
|
||||
hx-swap="outerHTML">
|
||||
{% set pas = 3 %}{% include '_stepper.html' %}
|
||||
<div class="card">
|
||||
<div style="display:flex; align-items:center; gap:8px; flex-wrap:wrap; margin-bottom:12px;">
|
||||
<h2 style="font-size:15px; margin:0;">
|
||||
<h2 style="font-size:var(--fs-md); margin:0;">
|
||||
Preview —
|
||||
<span class="muted" style="font-weight:400;">{{ filename or ("import #" ~ import_id) }}</span>
|
||||
</h2>
|
||||
<span class="muted" style="margin-left:auto; font-size:13px;">{{ total }} randuri</span>
|
||||
<span class="muted" style="margin-left:auto; font-size:var(--fs-sm);">{{ total }} randuri</span>
|
||||
</div>
|
||||
|
||||
{% if message %}
|
||||
@@ -17,113 +24,139 @@
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<!-- Rezumat stari (id stabil pentru OOB swap dupa editarea unui rand — US-002) -->
|
||||
<!-- Rezumat stari cu etichete umane cu majuscula (id stabil pentru OOB swap) -->
|
||||
{% set status_labels = [
|
||||
('ok', 'gata de trimis'),
|
||||
('needs_review', 'verifica valori'),
|
||||
('needs_mapping', 'fara cod RAR'),
|
||||
('needs_data', 'date lipsa'),
|
||||
('already_sent', 'deja trimis'),
|
||||
('duplicate_in_file','dublicat in fisier'),
|
||||
('ok', 'Gata de trimis'),
|
||||
('needs_review', 'Verifica valori'),
|
||||
('needs_mapping', 'Cod RAR lipsa'),
|
||||
('needs_data', 'Date incomplete'),
|
||||
('already_sent', 'Deja trimis'),
|
||||
('duplicate_in_file','Duplicat in fisier'),
|
||||
] %}
|
||||
<div id="preview-rezumat" style="display:flex; gap:8px; flex-wrap:wrap; margin-bottom:12px;">
|
||||
{% for status_key, label in status_labels %}
|
||||
{%- set cnt = summary.get(status_key, 0) -%}
|
||||
{% if cnt > 0 %}
|
||||
<span class="pill s-{{ status_key }}">{{ cnt }} {{ label }}</span>
|
||||
<span class="pill s-{{ status_key }}" style="display:inline-flex; align-items:center; gap:5px; font-size:var(--fs-xs);">
|
||||
<span aria-hidden="true" style="display:inline-block; width:7px; height:7px; border-radius:99px; background:currentColor; flex-shrink:0;"></span>{{ cnt }} {{ label | lower }}</span>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
||||
<!-- Butoane filtrare stare -->
|
||||
<!-- Butoane filtrare stare — text uman, data-filter pastreaza codul tehnic -->
|
||||
<div style="display:flex; gap:6px; flex-wrap:wrap; margin-bottom:10px;" role="group"
|
||||
aria-label="Filtrare dupa stare">
|
||||
<button type="button" class="filter-btn" data-filter="all"
|
||||
style="min-height:36px; font-size:13px; padding:4px 12px;">
|
||||
style="min-height:36px; font-size:var(--fs-sm); padding:4px 12px;">
|
||||
Toate ({{ total }})
|
||||
</button>
|
||||
{% for status_key, label in status_labels %}
|
||||
{%- set cnt = summary.get(status_key, 0) -%}
|
||||
{% if cnt > 0 %}
|
||||
<button type="button" class="filter-btn" data-filter="{{ status_key }}"
|
||||
style="min-height:36px; font-size:13px; padding:4px 12px;
|
||||
style="min-height:36px; font-size:var(--fs-sm); padding:4px 12px;
|
||||
background:transparent; border-color:var(--line); color:var(--ink);">
|
||||
{{ status_key }} ({{ cnt }})
|
||||
{{ label }} ({{ cnt }})
|
||||
</button>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
||||
<!-- Panou inline: operatii fara cod RAR, mapabile in flux (fara re-upload) -->
|
||||
<!-- Panou inline: operatii fara cod RAR, mapabile in flux (fara re-upload).
|
||||
US-004: un singur <form> cu un select per operatie + un singur buton Salveaza. -->
|
||||
{% if unmapped_ops %}
|
||||
<div class="card" style="border-color:var(--err); background:color-mix(in srgb, var(--err) 12%, var(--card)); margin-bottom:14px;">
|
||||
<h3 style="font-size:14px; margin:0 0 6px;">Operatii de mapat la cod RAR</h3>
|
||||
<p class="muted" style="margin:0 0 12px; font-size:13px;">
|
||||
<p class="muted" style="margin:0 0 12px; font-size:var(--fs-sm);">
|
||||
Aceste operatii din fisier nu au inca un cod RAR. Alege codul (sugestia e
|
||||
preselectata) si salveaza — randurile blocate trec automat in
|
||||
<span class="s-ok">ok</span> si maparea se retine pentru fisierele viitoare.
|
||||
</p>
|
||||
{% for e in unmapped_ops %}
|
||||
{%- set top = e.suggestions[0] if e.suggestions else None -%}
|
||||
{%- set preselect = top.cod_prestatie if (top and top.score >= 60) else '' -%}
|
||||
<form class="maprow" hx-post="/_import/{{ import_id }}/mapare-operatie"
|
||||
hx-target="#import-section" hx-swap="outerHTML"
|
||||
style="align-items:flex-end;">
|
||||
<form hx-post="/_import/{{ import_id }}/mapare-operatii"
|
||||
hx-target="#import-section" hx-swap="outerHTML">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<input type="hidden" name="cod_op_service" value="{{ e.cod_op_service }}">
|
||||
<div class="mapcol grow">
|
||||
<div><strong>{{ e.cod_op_service }}</strong>
|
||||
<span class="pill" title="randuri blocate">{{ e.blocked }} randuri</span></div>
|
||||
{% if e.denumire and e.denumire != e.cod_op_service %}
|
||||
<div class="muted">{{ e.denumire }}</div>
|
||||
{% endif %}
|
||||
{% if e.suggestions %}
|
||||
<div class="muted" style="font-size:12px; margin-top:4px;">
|
||||
sugestii:
|
||||
{% for s in e.suggestions[:3] %}
|
||||
<span class="sugg">{{ s.cod_prestatie }} ({{ s.score|round|int }}%)</span>{% if not loop.last %}, {% endif %}
|
||||
{% endfor %}
|
||||
{% for e in unmapped_ops %}
|
||||
{%- set top = e.suggestions[0] if e.suggestions else None -%}
|
||||
{%- set preselect = top.cod_prestatie if (top and top.score >= 60) else '' -%}
|
||||
<div class="maprow" style="align-items:flex-end; margin-bottom:10px;">
|
||||
<input type="hidden" name="cod_op_service" value="{{ e.cod_op_service }}">
|
||||
<div class="mapcol grow">
|
||||
<div><strong>{{ e.cod_op_service }}</strong>
|
||||
<span class="pill" title="randuri blocate">{{ e.blocked }} randuri</span></div>
|
||||
{% if e.denumire and e.denumire != e.cod_op_service %}
|
||||
<div class="muted">{{ e.denumire }}</div>
|
||||
{% endif %}
|
||||
{% if e.suggestions %}
|
||||
<div class="muted" style="font-size:12px; margin-top:4px;">
|
||||
sugestii:
|
||||
{% for s in e.suggestions[:3] %}
|
||||
<span class="sugg">{{ s.cod_prestatie }} ({{ s.score|round|int }}%)</span>{% if not loop.last %}, {% endif %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
<div class="mapcol">
|
||||
<select name="cod_prestatie" aria-label="Cod RAR pentru {{ e.cod_op_service }}">
|
||||
<option value="">— alege cod RAR —</option>
|
||||
{% for n in nomenclator %}
|
||||
<option value="{{ n.cod_prestatie }}" {% if n.cod_prestatie == preselect %}selected{% endif %}>
|
||||
{{ n.cod_prestatie }} — {{ n.nume_prestatie }}
|
||||
</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
<div class="mapcol">
|
||||
<select name="cod_prestatie" required aria-label="Cod RAR pentru {{ e.cod_op_service }}">
|
||||
<option value="">— alege cod RAR —</option>
|
||||
{% for n in nomenclator %}
|
||||
<option value="{{ n.cod_prestatie }}" {% if n.cod_prestatie == preselect %}selected{% endif %}>
|
||||
{{ n.cod_prestatie }} — {{ n.nume_prestatie }}
|
||||
</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</div>
|
||||
<div class="mapcol">
|
||||
{{ ui.autosend_toggle(checked=True, label="In coada automat") }}
|
||||
</div>
|
||||
<div class="mapcol">
|
||||
<button type="submit" style="min-height:44px;">Salveaza</button>
|
||||
{% endfor %}
|
||||
<div style="margin-top:12px;">
|
||||
<button type="submit" style="min-height:44px;">Salveaza maparile</button>
|
||||
</div>
|
||||
</form>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<!-- Tabel preview. Randurile au FORM PROPRIU pentru editare (NU sunt in #confirm-form,
|
||||
altfel Enter intr-un camp ar declansa trimiterea ireversibila — D-3.3). Bifele
|
||||
needs_review se asociaza la #confirm-form prin atributul form=. -->
|
||||
<div class="tablewrap">
|
||||
<!-- Banner discoverability (T1, US-007): vizibil cand exista randuri needs_review.
|
||||
Explica operatorului ca randurile cu 'Verifica valori' nu pleaca la RAR
|
||||
pana le deschide in modal si apasa 'Confirma valorile'. Dispare via OOB
|
||||
cand summary.needs_review == 0. -->
|
||||
<div id="preview-needs-review-banner">
|
||||
{% if summary.get('needs_review', 0) %}
|
||||
<div class="banner warn" role="note" aria-live="polite"
|
||||
style="margin-bottom:12px; padding:8px 14px; border-radius:6px;
|
||||
background:color-mix(in srgb, var(--warn, #e6b34a) 12%, var(--card));
|
||||
border:1px solid var(--warn, #e6b34a); font-size:13px;">
|
||||
Randurile cu <span class="pill s-needs_review" style="font-size:11px;">Verifica valori</span>
|
||||
nu pleaca la RAR pana le deschizi in modal si confirmi in modal
|
||||
cu butonul <strong>Confirma valorile</strong>.
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
<!-- Toggle randuri deja-trimise / duplicate: colapsate implicit (nu ocupa loc).
|
||||
Click -> comuta clasa .preview-arata-trimise pe tabel (CSS in base.html). -->
|
||||
{% set _n_trimise = summary.get('already_sent', 0) + summary.get('duplicate_in_file', 0) %}
|
||||
{% if _n_trimise %}
|
||||
<div style="margin-bottom:8px;">
|
||||
<button type="button" class="btn-secondary btn-sm" aria-expanded="false"
|
||||
onclick="var t=document.getElementById('preview-tabel'); var on=t.classList.toggle('preview-arata-trimise'); this.setAttribute('aria-expanded', on); this.querySelector('.tgl-tx').textContent = on ? 'Ascunde {{ _n_trimise }} deja trimise / duplicate' : 'Arata {{ _n_trimise }} deja trimise / duplicate';">
|
||||
<span class="tgl-tx">Arata {{ _n_trimise }} deja trimise / duplicate</span>
|
||||
</button>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<!-- Tabel preview in format identic cu tabelul Trimiteri (.tabel-trimiteri).
|
||||
5.16 (T-4): densitate redusa la coloanele esentiale — Stare / Vehicul /
|
||||
Operatie / Data + Actiuni. KM final + mesajul de validare (Note) au iesit
|
||||
din tabel: KM se editeaza in modal, motivul apare ca tooltip pe pill-ul de
|
||||
Stare. Randurile au FORM PROPRIU pentru editare (NU sunt in #confirm-form). -->
|
||||
<div id="preview-tabel" class="tablewrap tabel-trimiteri">
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
<th>#</th>
|
||||
<th>VIN</th>
|
||||
<th>Nr. Inm.</th>
|
||||
<th>Data</th>
|
||||
<th>KM final</th>
|
||||
<th>Operatie</th>
|
||||
<th>Stare</th>
|
||||
<th>Note</th>
|
||||
<th>Verificat?</th>
|
||||
<th>Actiuni</th>
|
||||
<th class="col-stare">Stare</th>
|
||||
<th class="col-vehicul">Vehicul</th>
|
||||
<th class="col-operatie">Operatie</th>
|
||||
<th class="col-data">Data</th>
|
||||
<th class="col-actiuni">Actiuni</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
@@ -132,6 +165,11 @@
|
||||
{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
<!-- Mesaj "filtrat la zero": afisat de JS cand filtrul ascunde toate randurile -->
|
||||
<p id="preview-zero-message" class="muted"
|
||||
style="display:none; text-align:center; padding:24px 16px; font-size:var(--fs-md);">
|
||||
Niciun rand nu corespunde filtrului selectat.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<!-- Bara confirmare (sticky jos) — singurul formular care trimite la RAR -->
|
||||
@@ -142,7 +180,7 @@
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
<div class="sticky-bar">
|
||||
<div style="flex:1; min-width:280px;">
|
||||
<!-- Banner declarant (D12) — direct deasupra input-ului N -->
|
||||
<!-- Banner declarant — direct deasupra input-ului N -->
|
||||
<div class="banner warn" style="margin-bottom:10px; padding:8px 12px; border-radius:6px;"
|
||||
role="note" aria-live="polite">
|
||||
Confirmand, TU esti declarantul acestor
|
||||
@@ -150,47 +188,32 @@
|
||||
prezentari la RAR (ireversibil).
|
||||
</div>
|
||||
|
||||
<div style="display:flex; gap:12px; align-items:center; flex-wrap:wrap;">
|
||||
<div>
|
||||
<label for="n-confirmat"
|
||||
style="font-size:13px; color:var(--muted); display:block; margin-bottom:2px;">
|
||||
Numar prezentari de confirmat
|
||||
</label>
|
||||
<input type="number" id="n-confirmat" name="n_confirmat"
|
||||
value="{{ summary.get('ok', 0) }}"
|
||||
min="0" required
|
||||
style="max-width:80px;"
|
||||
aria-describedby="n-hint">
|
||||
<span id="n-hint" class="muted" style="font-size:12px; margin-left:6px;">
|
||||
(<span id="n-hint-ok">{{ summary.get('ok', 0) }}</span> ok
|
||||
{% if summary.get('needs_review', 0) %}
|
||||
+ pana la {{ summary.get('needs_review', 0) }} verificate manual
|
||||
{% endif %})
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<label for="confirmed-by"
|
||||
style="font-size:13px; color:var(--muted); display:block; margin-bottom:2px;">
|
||||
Declarant (optional)
|
||||
</label>
|
||||
<input type="text" id="confirmed-by" name="confirmed_by"
|
||||
placeholder="email sau nume"
|
||||
style="max-width:200px;">
|
||||
</div>
|
||||
<div style="display:flex; gap:8px; align-items:center; flex-wrap:wrap;">
|
||||
<label for="n-confirmat"
|
||||
style="font-size:var(--fs-sm); color:var(--muted);">
|
||||
Confirma numarul
|
||||
</label>
|
||||
<input type="number" id="n-confirmat" name="n_confirmat"
|
||||
value="{{ summary.get('ok', 0) }}"
|
||||
min="0" required
|
||||
style="max-width:80px;"
|
||||
aria-describedby="n-hint">
|
||||
<span id="n-hint" class="muted" style="font-size:var(--fs-xs);">
|
||||
din <span id="n-hint-ok">{{ summary.get('ok', 0) }}</span> gata de trimis
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div style="display:flex; flex-direction:column; gap:6px; align-self:flex-end;">
|
||||
<button type="submit"
|
||||
id="confirm-btn"
|
||||
style="min-height:44px; padding:10px 28px; font-size:14px;"
|
||||
style="min-height:44px; padding:10px 28px; font-size:var(--fs-md);"
|
||||
{% if not summary.get('ok', 0) %}disabled title="Niciun rand ok de trimis"{% endif %}>
|
||||
Trimite la RAR
|
||||
</button>
|
||||
{% if summary.get('needs_data', 0) or summary.get('needs_mapping', 0) or summary.get('needs_review', 0) %}
|
||||
<a href="/v1/import/{{ import_id }}/export-failed" download
|
||||
style="font-size:12px; text-align:center;">
|
||||
style="font-size:var(--fs-xs); text-align:center;">
|
||||
descarca randuri cu probleme (CSV)
|
||||
</a>
|
||||
{% endif %}
|
||||
@@ -199,11 +222,11 @@
|
||||
</form>
|
||||
|
||||
<!-- Contor "gata de trimis" citit din DOM (data-ok), ca OOB swap-ul de la editare
|
||||
sa actualizeze N fara a re-randa sectiunea (US-002). -->
|
||||
sa actualizeze N fara a re-randa sectiunea. -->
|
||||
<span id="preview-ok-count" data-ok="{{ summary.get('ok', 0) }}" hidden></span>
|
||||
|
||||
<div style="padding:8px 0 4px;">
|
||||
<a href="#" class="muted" style="font-size:13px;"
|
||||
<a href="#" class="muted" style="font-size:var(--fs-sm);"
|
||||
hx-get="/_import/reset" hx-target="#import-section" hx-swap="outerHTML">Incarca alt fisier</a>
|
||||
</div>
|
||||
|
||||
@@ -212,39 +235,44 @@
|
||||
|
||||
<script>
|
||||
(function() {
|
||||
/* D-1.2: un singur sticky bar pe ecran — cat preview-ul de import e activ,
|
||||
/* Un singur sticky bar pe ecran — cat preview-ul de import e activ,
|
||||
ascunde sectiunea Trimiteri de pe Acasa (se reveleaza la reset/commit din _upload.html). */
|
||||
var trim = document.getElementById('trimiteri-section');
|
||||
if (trim) trim.style.display = 'none';
|
||||
|
||||
/* nOk se citeste din DOM (#preview-ok-count[data-ok]) ca OOB swap-ul de la editare
|
||||
sa-l poata actualiza fara re-randarea sectiunii (D-3.1/D-3.4). */
|
||||
sa-l poata actualiza fara re-randarea sectiunii. */
|
||||
function getOk() {
|
||||
var el = document.getElementById('preview-ok-count');
|
||||
return el ? parseInt(el.dataset.ok || '0', 10) : 0;
|
||||
}
|
||||
|
||||
/* Actualizeaza N si bannerul cand se bifeaza needs_review SAU cand se editeaza un rand. */
|
||||
/* Actualizeaza N dupa editare/confirmare rand (OOB).
|
||||
US-007: reviewed_rows (checkboxe) eliminate; N = randurile ok din DB,
|
||||
actualizate via OOB (#preview-ok-count[data-ok]) dupa /confirma-review sau /editeaza. */
|
||||
function updateN() {
|
||||
var checked = document.querySelectorAll('input[name="reviewed_rows"]:checked').length;
|
||||
var total = getOk() + checked;
|
||||
var total = getOk();
|
||||
var inp = document.getElementById('n-confirmat');
|
||||
var disp = document.getElementById('n-display');
|
||||
var btn = document.getElementById('confirm-btn');
|
||||
/* Nu re-activa confirm cat un rand e in editare (mutual-exclusion D-3.2). */
|
||||
var editing = document.querySelector('tr[data-editing="1"]') !== null;
|
||||
if (inp) inp.value = total;
|
||||
if (disp) disp.textContent = total;
|
||||
var hintOk = document.getElementById('n-hint-ok');
|
||||
if (hintOk) hintOk.textContent = getOk();
|
||||
if (btn) btn.disabled = (total === 0) || editing;
|
||||
if (hintOk) hintOk.textContent = total;
|
||||
if (btn) btn.disabled = (total === 0);
|
||||
}
|
||||
|
||||
/* Filtrare randuri dupa stare */
|
||||
/* Filtrare randuri dupa stare.
|
||||
Cand niciun rand nu e vizibil, afiseaza mesajul #preview-zero-message. */
|
||||
function filterRows(status) {
|
||||
var visible = 0;
|
||||
document.querySelectorAll('tbody tr[data-status]').forEach(function(tr) {
|
||||
tr.style.display = (status === 'all' || tr.dataset.status === status) ? '' : 'none';
|
||||
var show = status === 'all' || tr.dataset.status === status;
|
||||
tr.style.display = show ? '' : 'none';
|
||||
if (show) visible++;
|
||||
});
|
||||
var zeroMsg = document.getElementById('preview-zero-message');
|
||||
if (zeroMsg) zeroMsg.style.display = (visible === 0) ? '' : 'none';
|
||||
document.querySelectorAll('.filter-btn').forEach(function(b) {
|
||||
var active = b.dataset.filter === status;
|
||||
b.style.background = active ? 'var(--accent)' : '';
|
||||
@@ -260,5 +288,18 @@
|
||||
/* Filtru implicit "Toate" activ la incarcare */
|
||||
filterRows('all');
|
||||
updateN();
|
||||
|
||||
/* Evidentiere rand dupa reincarcarea preview-ului (window.__randSalvat setat de
|
||||
listener-ul 'randSalvat' din base.html): scroll + flash, ca userul sa vada CARE
|
||||
rand s-a schimbat si sa nu ramana cu impresia ca "nu s-a intamplat nimic". */
|
||||
if (window.__randSalvat) {
|
||||
var d = window.__randSalvat; window.__randSalvat = null;
|
||||
var r = document.getElementById('preview-row-' + d.rowIndex);
|
||||
if (r) {
|
||||
r.scrollIntoView({block:'center', behavior:'smooth'});
|
||||
void r.offsetWidth;
|
||||
r.classList.add('rand-actualizat');
|
||||
}
|
||||
}
|
||||
})();
|
||||
</script>
|
||||
|
||||
@@ -1,150 +1,67 @@
|
||||
{#
|
||||
_preview_rand.html — un singur rand de preview import (US-002, 3.6).
|
||||
Doua moduri:
|
||||
- display (editing falsy): <tr> normal + buton "Editeaza" pe coloana de actiuni.
|
||||
- edit (editing truthy): <tr> cu un singur <td colspan> ce contine un FORM PROPRIU
|
||||
(NU #confirm-form) cu grila responsiva refolosita din _trimitere_detaliu.html.
|
||||
Swap pe RAND (hx-target pe #preview-row-N, outerHTML), NU pe #import-section (D-3.1).
|
||||
La save, optional OOB pe rezumat + contor "gata de trimis" (include_oob).
|
||||
_preview_rand.html — un singur rand de preview import.
|
||||
US-006 (PRD 5.12): editarea inline (tr.preview-edit + mutual-exclusion script)
|
||||
a fost eliminata. Butonul Editeaza deschide MODALUL global (#detaliu-modal-body).
|
||||
|
||||
Parametri:
|
||||
editing — ELIMINAT (ignorat, pastrat pentru compatibilitate apeluri vechi)
|
||||
include_oob — True: randeaza OOB rezumat + contor + script recalc (swap dupa save)
|
||||
oob_tr — True: adauga hx-swap-oob pe <tr> insusi (pentru raspunsul POST succes)
|
||||
summary — dict cu contoarele per status
|
||||
|
||||
Campuri pre-computate de _web_compute_preview (NOT din template raw):
|
||||
row.prez — prezentare_din_payload(resolved): vehicul_nr, vin_scurt,
|
||||
operatie, cod_rar, data_prestatie, odometru
|
||||
row.stare_eticheta — text uman (ex. "Gata de trimis"), din STARI_PREVIEW
|
||||
row.stare_css — clasa CSS (ex. "s-ok"), din STARI_PREVIEW
|
||||
row.nota_umana — mesaj uman formatat pentru coloana Note (fara repr Python)
|
||||
#}
|
||||
{%- set res = row.resolved -%}
|
||||
{%- set status = row.resolved_status -%}
|
||||
{%- set prestatii = res.get('prestatii') or [] -%}
|
||||
{%- set op = (prestatii[0].get('cod_prestatie') or prestatii[0].get('cod_op_service', '')) if prestatii else '' -%}
|
||||
{% if editing %}
|
||||
{%- set err_map = {} -%}
|
||||
{%- set fix_map = {} -%}
|
||||
{%- for e in row.errors -%}{%- if e is mapping and e.get('field') -%}{%- set _ = err_map.update({e.field: (e.get('message') or e.get('msg'))}) -%}{%- if e.get('fix') -%}{%- set _ = fix_map.update({e.field: e.get('fix')}) -%}{%- endif -%}{%- endif -%}{%- endfor -%}
|
||||
<tr id="preview-row-{{ row.row_index }}" data-status="{{ status }}" data-editing="1">
|
||||
<td colspan="10" style="background:rgba(91,141,239,.06);">
|
||||
<form class="rand-editare"
|
||||
hx-post="/_import/{{ import_id }}/rand/{{ row.row_index }}/editeaza"
|
||||
hx-target="#preview-row-{{ row.row_index }}"
|
||||
hx-swap="outerHTML"
|
||||
hx-indicator="#rand-spinner-{{ row.row_index }}"
|
||||
hx-disabled-elt="find button"
|
||||
hx-on::response-error="this.querySelector('.rand-eroare-banner').style.display='block';">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token or '' }}">
|
||||
|
||||
<div style="display:flex; align-items:center; gap:8px; margin-bottom:8px;">
|
||||
<strong style="font-size:13px;">Editare rand {{ row.row_index + 1 }}</strong>
|
||||
<span class="pill s-{{ status }}" style="font-size:11px;">{{ status }}</span>
|
||||
</div>
|
||||
|
||||
{% if message %}
|
||||
<div class="flash" style="border-color:var(--err); background:color-mix(in srgb, var(--err) 12%, var(--card)); margin-bottom:10px;"
|
||||
role="alert">{{ message }}</div>
|
||||
{% endif %}
|
||||
<div class="rand-eroare-banner" role="alert"
|
||||
style="display:none; margin-bottom:10px; padding:8px 12px; border:1px solid var(--err);
|
||||
background:color-mix(in srgb, var(--err) 12%, var(--card)); border-radius:6px; font-size:13px;">
|
||||
Salvarea nu a reusit (retea / sesiune). Valorile introduse sunt pastrate — reincearca.
|
||||
</div>
|
||||
|
||||
{% macro camp(nume, eticheta, valoare, tip='text') %}
|
||||
<div>
|
||||
<label for="e-{{ row.row_index }}-{{ nume }}" class="muted" style="font-size:12px; display:block;">{{ eticheta }}</label>
|
||||
<input id="e-{{ row.row_index }}-{{ nume }}" type="{{ tip }}" name="{{ nume }}" value="{{ valoare or '' }}"
|
||||
style="width:100%; {% if err_map.get(nume) %}border-color:var(--err);{% endif %}"
|
||||
aria-label="{{ eticheta }} — randul {{ row.row_index + 1 }} (VIN: {{ res.get('vin') or '' }})"
|
||||
{% if err_map.get(nume) %}aria-invalid="true"{% endif %}>
|
||||
{% if err_map.get(nume) %}
|
||||
<div class="s-error" style="font-size:12px; margin-top:2px;">{{ err_map.get(nume) }}</div>
|
||||
{% endif %}
|
||||
{% if fix_map.get(nume) %}
|
||||
<span class="camp-fix">{{ fix_map.get(nume) }}</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endmacro %}
|
||||
|
||||
<div style="display:grid; grid-template-columns:repeat(auto-fit, minmax(200px, 1fr)); gap:10px 16px;">
|
||||
{{ camp('nr_inmatriculare', 'Numar inmatriculare', res.get('nr_inmatriculare')) }}
|
||||
{{ camp('vin', 'VIN (serie sasiu)', res.get('vin')) }}
|
||||
{{ camp('data_prestatie', 'Data prestatie (YYYY-MM-DD)', res.get('data_prestatie')) }}
|
||||
{{ camp('odometru_final', 'Odometru final', res.get('odometru_final')) }}
|
||||
{{ camp('odometru_initial', 'Odometru initial (daca e cerut)', res.get('odometru_initial')) }}
|
||||
</div>
|
||||
|
||||
<div style="margin-top:10px; display:flex; gap:8px; align-items:center; flex-wrap:wrap;">
|
||||
<button type="submit" style="min-height:44px; padding:8px 18px;">Salveaza</button>
|
||||
<button type="button" style="min-height:44px; padding:8px 18px;
|
||||
background:var(--card); color:var(--muted); border-color:var(--line);"
|
||||
hx-get="/_import/{{ import_id }}/rand/{{ row.row_index }}"
|
||||
hx-target="#preview-row-{{ row.row_index }}" hx-swap="outerHTML">Anuleaza</button>
|
||||
<span id="rand-spinner-{{ row.row_index }}" class="htmx-indicator muted" style="font-size:13px;">
|
||||
se salveaza…
|
||||
</span>
|
||||
</div>
|
||||
</form>
|
||||
</td>
|
||||
</tr>
|
||||
<script>
|
||||
(function() {
|
||||
/* Mutual-exclusion (D-3.2/3.6): cat un rand e in editare, dezactiveaza confirm + alte Editeaza. */
|
||||
var btn = document.getElementById('confirm-btn');
|
||||
if (btn) { btn.disabled = true; btn.title = 'Termina editarea randului inainte de a trimite.'; }
|
||||
document.querySelectorAll('.btn-editeaza').forEach(function(b) { b.disabled = true; });
|
||||
})();
|
||||
</script>
|
||||
{% else %}
|
||||
{%- set disp_fix_map = {} -%}
|
||||
{%- for e in row.errors -%}{%- if e is mapping and e.get('field') and e.get('fix') -%}{%- set _ = disp_fix_map.update({e.field: e.get('fix')}) -%}{%- endif -%}{%- endfor -%}
|
||||
{%- set _sent_dup = status in ('already_sent', 'duplicate_in_file') -%}
|
||||
<tr id="preview-row-{{ row.row_index }}" data-status="{{ status }}"
|
||||
style="{% if status == 'needs_review' %}background:rgba(230,179,74,.04);{% elif status in ('already_sent', 'duplicate_in_file') %}opacity:.65;{% endif %}">
|
||||
<td class="muted">{{ row.row_index + 1 }}</td>
|
||||
<td>{{ res.get('vin') or '<span class="muted">—</span>' | safe }}
|
||||
{% if disp_fix_map.get('vin') %}<span class="camp-fix">{{ disp_fix_map.get('vin') }}</span>{% endif %}
|
||||
{% if _sent_dup %}class="preview-sent-row"{% endif %}
|
||||
{% if oob_tr %}hx-swap-oob="outerHTML:#preview-row-{{ row.row_index }}"{% endif %}
|
||||
style="{% if status == 'needs_review' %}background:rgba(230,179,74,.04);{% elif _sent_dup %}opacity:.6;{% endif %}">
|
||||
{#- Motivul (validare / deja-trimis / duplicat) — fost coloana Note, acum tooltip pe pill.
|
||||
KM final iese din tabel (se editeaza in modal). -#}
|
||||
{%- if status == 'already_sent' and row.get('already_sent_info') -%}
|
||||
{%- set ai = row.already_sent_info -%}
|
||||
{%- set _nota = 'deja trimis ' ~ ((ai.get('created_at') or '')[:10]) ~ ((' (#' ~ ai.id_prezentare ~ ')') if ai.get('id_prezentare') else '') -%}
|
||||
{%- elif status == 'duplicate_in_file' and row.get('duplicate_with') -%}
|
||||
{%- set _dwith = [] -%}
|
||||
{%- for idx in row.duplicate_with -%}{{ _dwith.append(idx + 1) or '' }}{%- endfor -%}
|
||||
{%- set _nota = 'dubla cu randul ' ~ (_dwith | join(', ')) -%}
|
||||
{%- else -%}
|
||||
{%- set _nota = row.nota_umana or '' -%}
|
||||
{%- endif -%}
|
||||
<td class="col-stare" data-eticheta="Stare">
|
||||
<span class="pill {{ row.stare_css }}" style="display:inline-flex; align-items:center; gap:5px;"
|
||||
{% if _nota %}title="{{ _nota }}"{% endif %}>
|
||||
<span aria-hidden="true" style="display:inline-block; width:7px; height:7px; border-radius:99px; background:currentColor; flex-shrink:0;"></span>{{ row.stare_eticheta }}</span>
|
||||
</td>
|
||||
<td>{{ res.get('nr_inmatriculare') or '' }}
|
||||
{% if disp_fix_map.get('nr_inmatriculare') %}<span class="camp-fix">{{ disp_fix_map.get('nr_inmatriculare') }}</span>{% endif %}
|
||||
</td>
|
||||
<td>{{ res.get('data_prestatie') or '' }}
|
||||
{% if disp_fix_map.get('data_prestatie') %}<span class="camp-fix">{{ disp_fix_map.get('data_prestatie') }}</span>{% endif %}
|
||||
</td>
|
||||
<td>{{ res.get('odometru_final') or '' }}
|
||||
{% if disp_fix_map.get('odometru_final') %}<span class="camp-fix">{{ disp_fix_map.get('odometru_final') }}</span>{% endif %}
|
||||
</td>
|
||||
<td>{{ op or '<span class="muted">—</span>' | safe }}</td>
|
||||
<td><span class="pill s-{{ status }}">{{ status }}</span></td>
|
||||
<td class="muted" style="font-size:12px; white-space:normal; max-width:220px;">
|
||||
{% if status == 'already_sent' and row.get('already_sent_info') %}
|
||||
{% set ai = row.already_sent_info %}
|
||||
deja trimis {{ (ai.get('created_at') or '')[:10] }}
|
||||
{% if ai.get('id_prezentare') %}(#{{ ai.id_prezentare }}){% endif %}
|
||||
{% elif status == 'duplicate_in_file' and row.get('duplicate_with') %}
|
||||
dubla cu randul
|
||||
{% for idx in row.duplicate_with %}{{ idx + 1 }}{% if not loop.last %}, {% endif %}{% endfor %}
|
||||
{% elif row.flags %}
|
||||
{{ row.flags[0] }}
|
||||
{% elif row.errors %}
|
||||
{%- for e in row.errors -%}
|
||||
{%- if e is mapping -%}
|
||||
{{ e.get('message') or e.get('msg') or (e.values() | list | first) }}
|
||||
{%- else -%}
|
||||
{{ e }}
|
||||
{%- endif -%}
|
||||
{%- if not loop.last %}; {% endif -%}
|
||||
{%- endfor -%}
|
||||
<td class="col-vehicul" data-eticheta="Vehicul">
|
||||
{{ row.prez.vehicul_nr }}
|
||||
{% if row.prez.vin_scurt and row.prez.vin_scurt != '—' %}
|
||||
<div class="muted" style="font-size:12px; white-space:nowrap;">{{ row.prez.vin_scurt }}</div>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td style="text-align:center;">
|
||||
{% if status == 'needs_review' %}
|
||||
<label class="chk" style="min-height:44px; justify-content:center; cursor:pointer;"
|
||||
title="Bifat inseamna ca ai verificat valorile si le incluzi in trimitere">
|
||||
<input type="checkbox" form="confirm-form" name="reviewed_rows" value="{{ row.row_index }}"
|
||||
onchange="window.updateN && window.updateN()"
|
||||
aria-label="Verificat — randul {{ row.row_index + 1 }} (VIN: {{ res.get('vin', '') }})">
|
||||
verif.
|
||||
</label>
|
||||
<td class="col-operatie" data-eticheta="Operatie">
|
||||
<div>{{ row.prez.operatie }}</div>
|
||||
{% if row.prez.cod_rar and row.prez.cod_rar != '—' %}
|
||||
<div class="cod-rar-sub"><span class="cod-rar-cod">{{ row.prez.cod_rar }}</span></div>
|
||||
{% else %}
|
||||
<div class="muted cod-rar-sub">nemapat</div>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td style="text-align:center;">
|
||||
<td class="col-data" data-eticheta="Data prestatie">{{ row.prez.data_prestatie }}</td>
|
||||
<td class="col-actiuni" data-eticheta="Actiuni" style="text-align:center;">
|
||||
{% if status not in ('already_sent', 'duplicate_in_file') %}
|
||||
<button type="button" class="btn-editeaza"
|
||||
style="min-height:44px; padding:6px 14px; font-size:13px;
|
||||
style="min-height:36px; padding:6px 14px; font-size:13px;
|
||||
background:transparent; border-color:var(--line); color:var(--ink);"
|
||||
hx-get="/_import/{{ import_id }}/rand/{{ row.row_index }}/editare"
|
||||
hx-target="#preview-row-{{ row.row_index }}" hx-swap="outerHTML"
|
||||
hx-get="/_import/{{ import_id }}/rand/{{ row.row_index }}/editare-modal"
|
||||
hx-target="#detaliu-modal-body" hx-swap="innerHTML"
|
||||
aria-label="Editeaza randul {{ row.row_index + 1 }} (VIN: {{ res.get('vin', '') }})">
|
||||
Editeaza
|
||||
</button>
|
||||
@@ -152,18 +69,31 @@
|
||||
</td>
|
||||
</tr>
|
||||
{% if include_oob %}
|
||||
{# OOB: actualizeaza rezumatul si contorul "gata de trimis" dupa save, fara a re-randa sectiunea (D-3.1). #}
|
||||
{# OOB: actualizeaza rezumatul, contorul, bannerul needs_review dupa save/confirma-review. #}
|
||||
{% set status_labels = [
|
||||
('ok','gata de trimis'), ('needs_review','verifica valori'), ('needs_mapping','fara cod RAR'),
|
||||
('needs_data','date lipsa'), ('already_sent','deja trimis'), ('duplicate_in_file','dublicat in fisier')] %}
|
||||
('ok','Gata de trimis'), ('needs_review','Verifica valori'), ('needs_mapping','Cod RAR lipsa'),
|
||||
('needs_data','Date incomplete'), ('already_sent','Deja trimis'), ('duplicate_in_file','Duplicat in fisier')] %}
|
||||
<div id="preview-rezumat" hx-swap-oob="true"
|
||||
style="display:flex; gap:8px; flex-wrap:wrap; margin-bottom:12px;">
|
||||
{% for status_key, label in status_labels %}
|
||||
{%- set cnt = summary.get(status_key, 0) -%}
|
||||
{% if cnt > 0 %}<span class="pill s-{{ status_key }}">{{ cnt }} {{ label }}</span>{% endif %}
|
||||
{% if cnt > 0 %}<span class="pill s-{{ status_key }}" style="display:inline-flex; align-items:center; gap:5px; font-size:var(--fs-xs);"><span aria-hidden="true" style="display:inline-block; width:7px; height:7px; border-radius:99px; background:currentColor; flex-shrink:0;"></span>{{ cnt }} {{ label | lower }}</span>{% endif %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
<span id="preview-ok-count" hx-swap-oob="true" data-ok="{{ summary.get('ok', 0) }}" hidden></span>
|
||||
{# Banner discoverability: OOB swap dupa confirmare/editare → dispare cand needs_review==0. #}
|
||||
<div id="preview-needs-review-banner" hx-swap-oob="true">
|
||||
{% if summary.get('needs_review', 0) %}
|
||||
<div class="banner warn" role="note" aria-live="polite"
|
||||
style="margin-bottom:12px; padding:8px 14px; border-radius:6px;
|
||||
background:color-mix(in srgb, var(--warn, #e6b34a) 12%, var(--card));
|
||||
border:1px solid var(--warn, #e6b34a); font-size:13px;">
|
||||
Randurile cu <span class="pill s-needs_review" style="font-size:11px;">Verifica valori</span>
|
||||
nu pleaca la RAR pana le deschizi in modal si confirmi in modal
|
||||
cu butonul <strong>Confirma valorile</strong>.
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
<script>
|
||||
(function() {
|
||||
/* Editare incheiata: re-activeaza confirm + butoanele Editeaza, recalculeaza N.
|
||||
@@ -179,4 +109,3 @@
|
||||
})();
|
||||
</script>
|
||||
{% endif %}
|
||||
{% endif %}
|
||||
|
||||
@@ -1,9 +1,10 @@
|
||||
<div id="status-bar" class="status-bar card"
|
||||
hx-get="/_fragments/status"
|
||||
hx-trigger="every 15s"
|
||||
hx-swap="outerHTML">
|
||||
hx-get="/_fragments/status?tab={{ tab_activ | default('acasa') }}"
|
||||
hx-trigger="every 15s, trimiteriChanged from:body"
|
||||
hx-swap="outerHTML"
|
||||
{% if oob %}hx-swap-oob="outerHTML"{% endif %}>
|
||||
|
||||
<!-- Cont in asteptare de activare (regasit din vechiul _banner; mereu vizibil) -->
|
||||
{# Banner cont in asteptare de activare (mereu vizibil cand contul e inactiv) #}
|
||||
{% if not account_active %}
|
||||
<div style="margin-bottom:12px; padding:8px 10px; border-left:3px solid var(--warn);
|
||||
background:color-mix(in srgb, var(--warn) 12%, var(--card)); border-radius:6px; font-size:13px;">
|
||||
@@ -13,71 +14,143 @@
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
<!-- Rand 1: doua bife binare + ultima autentificare -->
|
||||
<div style="display:flex; gap:28px; flex-wrap:wrap; align-items:center; font-size:14px;">
|
||||
|
||||
{# Bifa: glifa (✓/✗) + culoare + text — accesibil (nu doar culoare, design review) #}
|
||||
{% macro bifa(ok, text, tip) %}
|
||||
<span title="{{ tip }}" style="display:inline-flex; align-items:center; gap:7px;">
|
||||
{% if ok %}
|
||||
<span class="s-sent" aria-hidden="true" style="font-weight:bold;">✓</span>
|
||||
<span class="s-sent">{{ text }}</span>
|
||||
{% else %}
|
||||
<span class="s-error" aria-hidden="true" style="font-weight:bold;">✗</span>
|
||||
<span class="s-error">{{ text }}</span>
|
||||
{% endif %}
|
||||
</span>
|
||||
{% endmacro %}
|
||||
|
||||
{{ bifa(worker_ok, worker_lbl[0], worker_lbl[1]) }}
|
||||
{{ bifa(rar_ok, rar_lbl[0], rar_lbl[1]) }}
|
||||
|
||||
<span style="display:inline-flex; align-items:center; gap:6px;">
|
||||
<span class="muted">{{ eticheta_ultima_auth }}:</span>
|
||||
<span>{{ last_login }}</span>
|
||||
</span>
|
||||
{# US-006 (5.17) — Banner one-time trial->Gratuit (T-DES-1): afisat la prima incarcare
|
||||
dupa expirarea trial-ului. Discret, non-blocant; dismissibil via sessionStorage.
|
||||
Nu acopera stripul de sanatate (apare inainte de health strip, la acelasi nivel). #}
|
||||
{% if trial_expirat_recent|default(false) %}
|
||||
<div id="banner-trial-expirat"
|
||||
role="status"
|
||||
style="margin-bottom:10px; padding:7px 12px;
|
||||
border-left:3px solid var(--warn);
|
||||
background:color-mix(in srgb, var(--warn) 10%, var(--card));
|
||||
border-radius:6px; font-size:var(--fs-sm);
|
||||
display:flex; align-items:center; justify-content:space-between; gap:8px;">
|
||||
<span>Trial Pro expirat — esti pe Gratuit, 60/luna</span>
|
||||
<button onclick="sessionStorage.setItem('tfx','1'); document.getElementById('banner-trial-expirat').style.display='none';"
|
||||
style="background:transparent; border:none; color:var(--muted); cursor:pointer;
|
||||
font-size:18px; padding:0 4px; line-height:1; flex-shrink:0;"
|
||||
aria-label="Inchide bannerul">×</button>
|
||||
</div>
|
||||
<script>(function(){ if(sessionStorage.getItem('tfx')){ var el=document.getElementById('banner-trial-expirat'); if(el) el.style.display='none'; } })();</script>
|
||||
{% endif %}
|
||||
|
||||
<!-- Rand 2: contoare coada -->
|
||||
<div style="margin-top:10px; display:flex; gap:20px; flex-wrap:wrap; font-size:14px;">
|
||||
<span><span class="muted">In asteptare:</span> <span class="s-queued">{{ counts_queued }}</span></span>
|
||||
<span><span class="muted">Declarate la RAR:</span> <span class="s-sent">{{ counts_sent }}</span></span>
|
||||
<span><span class="muted">Blocate:</span>
|
||||
<span class="{{ 's-error' if blocate_total else 'muted' }}">{{ blocate_total }}</span>
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<!-- Necesita atentia ta (US-014): categorii actionabile — link la lista filtrata
|
||||
+ identificatorii primelor randuri blocate. Se randeaza DOAR daca exista randuri
|
||||
blocate; cand contorul ajunge 0 (sters/re-pus/purjat), sectiunea dispare. -->
|
||||
{% if blocate_actionabil %}
|
||||
<div style="margin-top:12px; padding-top:10px; border-top:1px solid var(--line);">
|
||||
<div style="font-size:13px; font-weight:600; margin-bottom:8px;">Necesita atentia ta</div>
|
||||
<div style="display:flex; gap:18px; flex-wrap:wrap;">
|
||||
{% for cat in blocate_actionabil %}
|
||||
<div style="min-width:200px;">
|
||||
{# Link: filtreaza lista Trimiteri pe aceasta stare (HTMX in-place) cu fallback
|
||||
deep-link server-side (?tab=acasa&status=...). #}
|
||||
<a class="{{ cat.eticheta[2] }}" style="font-size:13px; font-weight:600; text-decoration:none;"
|
||||
href="/?tab=acasa&status={{ cat.status }}"
|
||||
hx-get="/_fragments/submissions?status={{ cat.status }}"
|
||||
hx-target="#submissions-wrap" hx-swap="innerHTML"
|
||||
onclick="var s=document.getElementById('trimiteri-section'); if(s) s.scrollIntoView({behavior:'smooth'});">
|
||||
{{ cat.eticheta[0] }} ({{ cat.n }}) ›
|
||||
</a>
|
||||
<ul style="list-style:none; margin:6px 0 0; padding:0;">
|
||||
{% for r in cat.randuri %}
|
||||
<li class="muted" style="font-size:12px;">
|
||||
#{{ r.id }} {{ r.vin }}{% if r.nr %} / {{ r.nr }}{% endif %}
|
||||
</li>
|
||||
{% endfor %}
|
||||
{% if cat.rest %}
|
||||
<li class="muted" style="font-size:12px;">…si inca {{ cat.rest }}</li>
|
||||
{% endif %}
|
||||
</ul>
|
||||
</div>
|
||||
{% endfor %}
|
||||
{# === US-003 (PRD 5.16): Banda de stare RAR — NUMAI cand BLOCAT (rosu, lat de 100%).
|
||||
OK = dot verde in antet (base.html); banda nu mai apare cand totul e ok.
|
||||
Elementul id="strip-sanatate" ramane in DOM mereu, dar goleste continutul cand OK,
|
||||
astfel "hidden" + fara continut eroare in sursa = nu pica testele de prezenta id-ului.
|
||||
#}
|
||||
{% if sanatate_ok %}
|
||||
<div id="strip-sanatate" role="status" aria-live="polite" hidden></div>
|
||||
{% else %}
|
||||
<div id="strip-sanatate"
|
||||
role="status"
|
||||
aria-live="polite"
|
||||
style="display:flex; align-items:center; justify-content:space-between; gap:12px; flex-wrap:wrap;
|
||||
padding:10px 14px; border-radius:8px; margin-bottom:14px;
|
||||
background:color-mix(in srgb, var(--err) 16%, var(--card)); border:1px solid color-mix(in srgb, var(--err) 40%, transparent);">
|
||||
<div style="display:flex; align-items:center; gap:9px;">
|
||||
<span aria-hidden="true" style="font-weight:700; font-size:15px; color:var(--err);">✗</span>
|
||||
<span style="font-weight:700; font-size:13px;">{{ sanatate_text }}</span>
|
||||
</div>
|
||||
<span style="font-family:var(--font-mono); font-size:var(--fs-xs); color:var(--muted); white-space:nowrap;">
|
||||
{{ eticheta_ultima_auth }}: {{ last_login }}
|
||||
</span>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === US-002 (PRD 5.16): 5 carduri-contor separate (desktop) + bara compacta (mobil <=560px).
|
||||
Total / Luna asta / Azi / In coada / De corectat.
|
||||
#}
|
||||
{# Desktop: 5 carduri side-by-side. display:flex + layout stau in CSS (.contoare-desktop in
|
||||
base.html), NU inline, ca media query-ul <=560px sa le poata ascunde pe mobil (bara compacta). #}
|
||||
<div class="contoare-desktop">
|
||||
|
||||
{# Total trimise (all-time) #}
|
||||
<div class="contor-card" style="flex:1; min-width:100px;">
|
||||
<div class="contor-cifra">{{ counts_sent }}</div>
|
||||
<div class="contor-label">Total</div>
|
||||
</div>
|
||||
|
||||
{# Luna asta #}
|
||||
<div class="contor-card" style="flex:1; min-width:100px;">
|
||||
<div class="contor-cifra s-accent">{{ sent_month }}</div>
|
||||
<div class="contor-label">Luna asta</div>
|
||||
</div>
|
||||
|
||||
{# Azi #}
|
||||
<div class="contor-card" style="flex:1; min-width:80px;">
|
||||
<div class="contor-cifra s-accent">{{ sent_today }}</div>
|
||||
<div class="contor-label">Azi</div>
|
||||
</div>
|
||||
|
||||
{# In coada #}
|
||||
<div class="contor-card" style="flex:1; min-width:80px;">
|
||||
<div class="contor-cifra s-queued">{{ counts_queued }}</div>
|
||||
<div class="contor-label">In coada</div>
|
||||
</div>
|
||||
|
||||
{# De corectat (rosu daca >0, muted la 0; link catre lista) #}
|
||||
<a href="/" class="contor-card"
|
||||
style="flex:1; min-width:80px; text-decoration:none; display:block; cursor:pointer;"
|
||||
aria-label="De corectat: {{ blocate_total }} — click pentru lista de trimiteri">
|
||||
<div class="contor-cifra {{ 's-error' if blocate_total else 'muted' }}">{{ blocate_total }}</div>
|
||||
<div class="contor-label">De corectat</div>
|
||||
</a>
|
||||
|
||||
</div>
|
||||
|
||||
{# Mobil (<=560px): bara compacta — numerele + etichete scurte in-line #}
|
||||
<div class="contoare-compact">
|
||||
<div class="compact-item">
|
||||
<div class="compact-nr">{{ counts_sent }}</div>
|
||||
<div class="compact-lbl">Total</div>
|
||||
</div>
|
||||
<div class="compact-item">
|
||||
<div class="compact-nr s-accent">{{ sent_month }}</div>
|
||||
<div class="compact-lbl">Luna</div>
|
||||
</div>
|
||||
<div class="compact-item">
|
||||
<div class="compact-nr s-accent">{{ sent_today }}</div>
|
||||
<div class="compact-lbl">Azi</div>
|
||||
</div>
|
||||
<div class="compact-item">
|
||||
<div class="compact-nr s-queued">{{ counts_queued }}</div>
|
||||
<div class="compact-lbl">Coada</div>
|
||||
</div>
|
||||
<a class="compact-item" href="/" style="text-decoration:none; color:inherit;">
|
||||
<div class="compact-nr {{ 's-error' if blocate_total else 'muted' }}">{{ blocate_total }}</div>
|
||||
<div class="compact-lbl">Erori</div>
|
||||
</a>
|
||||
</div>
|
||||
|
||||
{# === Navigatie rapida: Trimiteri + Mapari cu badge needs_mapping ===
|
||||
Pastrata exact ca inainte (US-005): tab_activ determina marcajul activ.
|
||||
#}
|
||||
{% set _tab = tab_activ | default('acasa') %}
|
||||
<nav class="status-nav" aria-label="Navigatie rapida"
|
||||
style="display:flex; gap:8px 16px; flex-wrap:wrap; font-size:13px; border-top:1px solid var(--line); padding-top:8px;">
|
||||
<a href="/"
|
||||
{% if _tab == 'acasa' or _tab == '' %}aria-current="page"{% endif %}
|
||||
class="status-nav-link{% if _tab == 'acasa' or _tab == '' %} status-nav-activ{% endif %}">Trimiteri</a>
|
||||
<a href="/?tab=mapari"
|
||||
{% if _tab == 'mapari' %}aria-current="page"{% endif %}
|
||||
class="status-nav-link{% if _tab == 'mapari' %} status-nav-activ{% endif %}">Mapari{% if mapari_badge %}<span class="tab-badge" aria-hidden="true" style="display:inline-flex; align-items:center; justify-content:center; min-width:16px; height:16px; margin-left:4px; padding:0 4px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;">{{ mapari_badge }}</span>{% endif %}</a>
|
||||
</nav>
|
||||
|
||||
{# US-006 (5.17) + T-6 (5.16): linia de plan in CORP apare DOAR in starea de avertizare
|
||||
(>=80% -> --warn; limita atinsa -> --err). Consumul normal (N/60) traieste in badge-ul
|
||||
din antet + linia din meniul burger, nu ca rand permanent in corp (densitate redusa).
|
||||
Ierarhie: nu concureaza cu stripul de sanatate (zero-silent-failures pastrat). #}
|
||||
{% if plan_linie and (plan_warn|default(false) or plan_limita_atinsa|default(false)) %}
|
||||
<div class="plan-status-line"
|
||||
style="font-size:var(--fs-sm); margin-top:6px; padding-top:6px;
|
||||
border-top:1px solid var(--line2);
|
||||
color:{% if plan_limita_atinsa|default(false) %}var(--err){% elif plan_warn|default(false) %}var(--warn){% else %}var(--muted){% endif %};
|
||||
{% if plan_warn|default(false) %}font-weight:600;{% endif %}">
|
||||
{{ plan_linie }}
|
||||
{% if plan_limita_atinsa|default(false) or plan_warn|default(false) %}
|
||||
<a href="/?tab=cont" style="font-size:var(--fs-xs); font-weight:400; color:var(--accent);">Detalii plan</a>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
|
||||
@@ -1,22 +1,9 @@
|
||||
{#
|
||||
_stepper.html — Antet wizard import (PUR vizual, fara logica de rutare).
|
||||
|
||||
Parametru: `pas` (integer 1-4) — pasul curent.
|
||||
Utilizare in template-uri care mostenesc contextul Jinja2:
|
||||
{% set pas = 1 %}{% include '_stepper.html' %}
|
||||
sau cu `with`:
|
||||
{% with pas=2 %}{% include '_stepper.html' %}{% endwith %}
|
||||
|
||||
Cei 4 pasi ficsi:
|
||||
1. Incarca fisier
|
||||
2. Potriveste coloanele
|
||||
3. Verifica
|
||||
4. Confirma trimiterea
|
||||
|
||||
Stari vizuale:
|
||||
- index < pas → "facut" (bulina plina, text bifat)
|
||||
- index == pas → "activ" (evidentiat, aria-current="step")
|
||||
- index > pas → "viitor" (estompat)
|
||||
_stepper.html — Antet wizard import COMPACT (PUR vizual). PRD 5.13.
|
||||
Parametru: `pas` (integer 1-4). Clasele .stepper-* sunt definite in base.html.
|
||||
>=1024px: bara slim orizontala (.stepper-track). <1024px: forma colapsata
|
||||
"Pasul N din 4 - Titlu" + bara de progres (.stepper-collapsed).
|
||||
Utilizare: {% set pas = 1 %}{% include '_stepper.html' %}
|
||||
#}
|
||||
{%- set _pasi_import = [
|
||||
(1, "Incarca fisier", "Trage un fisier xlsx/csv aici sau foloseste butonul de alegere."),
|
||||
@@ -24,73 +11,26 @@
|
||||
(3, "Verifica", "Verifica randurile inainte sa le trimiti la RAR."),
|
||||
(4, "Confirma trimiterea", "Confirma numarul de prezentari — actiunea e ireversibila."),
|
||||
] -%}
|
||||
<nav class="stepper-import" aria-label="Pasii importului" style="
|
||||
display:flex;
|
||||
gap:0;
|
||||
align-items:stretch;
|
||||
margin-bottom:20px;
|
||||
border:1px solid var(--line);
|
||||
border-radius:8px;
|
||||
overflow:hidden;
|
||||
background:var(--card);
|
||||
">
|
||||
{% for nr, titlu, ajutor in _pasi_import %}
|
||||
{%- if nr < pas %}
|
||||
{%- set cls = "facut" -%}
|
||||
{%- set aria = "" -%}
|
||||
{%- elif nr == pas %}
|
||||
{%- set cls = "activ" -%}
|
||||
{%- set aria = ' aria-current="step"' -%}
|
||||
{%- else %}
|
||||
{%- set cls = "viitor" -%}
|
||||
{%- set aria = "" -%}
|
||||
{%- endif %}
|
||||
<div class="stepper-pas stepper-pas--{{ cls }}"{{ aria | safe }}
|
||||
style="
|
||||
flex:1;
|
||||
padding:10px 14px;
|
||||
border-right:{% if not loop.last %}1px solid var(--line){% else %}none{% endif %};
|
||||
{% if cls == 'activ' %}
|
||||
background:rgba(91,141,239,.10);
|
||||
{% elif cls == 'facut' %}
|
||||
opacity:1;
|
||||
{% else %}
|
||||
opacity:.4;
|
||||
{% endif %}
|
||||
">
|
||||
<div style="display:flex; align-items:center; gap:6px; margin-bottom:2px;">
|
||||
<span class="stepper-nr" style="
|
||||
display:inline-flex;
|
||||
align-items:center;
|
||||
justify-content:center;
|
||||
width:20px;
|
||||
height:20px;
|
||||
border-radius:50%;
|
||||
font-size:11px;
|
||||
font-weight:700;
|
||||
flex-shrink:0;
|
||||
{% if cls == 'activ' %}
|
||||
background:var(--accent);
|
||||
color:#fff;
|
||||
{% elif cls == 'facut' %}
|
||||
background:var(--ok);
|
||||
color:#fff;
|
||||
{% else %}
|
||||
background:var(--line);
|
||||
color:var(--muted);
|
||||
{% endif %}
|
||||
">
|
||||
{% if cls == 'facut' %}✓{% else %}{{ nr }}{% endif %}
|
||||
</span>
|
||||
<span style="
|
||||
font-size:13px;
|
||||
font-weight:{% if cls == 'activ' %}600{% else %}400{% endif %};
|
||||
color:{% if cls == 'activ' %}var(--ink){% elif cls == 'facut' %}var(--ink){% else %}var(--muted){% endif %};
|
||||
">{{ titlu }}</span>
|
||||
{%- set _activ = _pasi_import[pas - 1] -%}
|
||||
<div class="stepper">
|
||||
{# Desktop (>=1024px): bara slim orizontala. #}
|
||||
<nav class="stepper-track" aria-label="Pasii importului">
|
||||
{% for nr, titlu, ajutor in _pasi_import %}
|
||||
{%- if nr < pas %}{% set cls = "is-done" %}{% set aria = "" %}
|
||||
{%- elif nr == pas %}{% set cls = "is-active" %}{% set aria = ' aria-current="step"' %}
|
||||
{%- else %}{% set cls = "" %}{% set aria = "" %}{% endif %}
|
||||
<div class="stepper-step {{ cls }}"{{ aria | safe }}>
|
||||
<span class="stepper-nr">{% if nr < pas %}✓{% else %}{{ nr }}{% endif %}</span>
|
||||
<span class="stepper-tx">{{ titlu }}</span>
|
||||
</div>
|
||||
{% if cls == 'activ' %}
|
||||
<p class="muted" style="margin:0; font-size:12px; padding-left:26px;">{{ ajutor }}</p>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</nav>
|
||||
{# Tableta/mobil (<1024px): colapsat "Pasul N din 4 - Titlu" + progres. #}
|
||||
<div class="stepper-collapsed">
|
||||
<div class="stepper-current">Pasul {{ pas }} din 4 <span class="muted">· {{ _activ[1] }}</span></div>
|
||||
<div class="stepper-progress" role="progressbar" aria-valuenow="{{ pas }}" aria-valuemin="1" aria-valuemax="4"
|
||||
aria-label="Pasul {{ pas }} din 4"><span style="width:{{ (pas / 4 * 100) | round | int }}%;"></span></div>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</nav>
|
||||
{# Ajutorul pasului activ — o singura linie, sub bara (valabil pe ambele forme). #}
|
||||
<p class="stepper-help">{{ _activ[2] }}</p>
|
||||
</div>
|
||||
|
||||
@@ -1,6 +1,33 @@
|
||||
{#
|
||||
OOB: actualizeaza inputul id="f-page" din #filtre-trimiteri.
|
||||
Reincarcarea (hx-include="#filtre-trimiteri") preia automat pagina curenta.
|
||||
Elementul OOB e extras din continutul normal de HTMX inainte de swap in #submissions-wrap.
|
||||
#}
|
||||
<input type="hidden" id="f-page" name="page" value="{{ page | default(1) }}" hx-swap-oob="true">
|
||||
|
||||
{# OOB: re-randeaza pill-urile de stare (in bara de filtre, in afara #submissions-wrap) cu
|
||||
contoarele si starea activa proaspete la fiecare reincarcare a tabelului. #}
|
||||
<span hx-swap-oob="innerHTML:#pills-categorii">{% include '_pills.html' %}</span>
|
||||
|
||||
{# Versiunea datelor cu care s-a randat tabelul; pollerul "Date noi" o compara. #}
|
||||
<span id="trimiteri-versiune" data-v="{{ versiune_trimiteri | default('') }}" hidden></span>
|
||||
|
||||
{% if bulk_message %}
|
||||
{# Sumar actiune bulk (US-010 PRD 5.15): afisat dupa bulk-fix, disparut la urmatoarea reincarcare. #}
|
||||
<div class="bulk-message" role="status" aria-live="polite"
|
||||
style="font-size:13px; color:var(--ink); background:var(--card2);
|
||||
border:1px solid var(--line); border-radius:6px;
|
||||
padding:6px 10px; margin-bottom:8px;">
|
||||
{{ bulk_message }}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{% if rows %}
|
||||
{# US-011: form de stergere bulk. Selectia opereaza DOAR pe randuri blocate
|
||||
(gestionabil); sent/sending/queued nu au checkbox (read-only). #}
|
||||
{# Form bulk cu DOUA actiuni: (1) aplica cod RAR la selectate (bulk-fix, US-010),
|
||||
(2) sterge selectate (sterge-bulk). Selectia opereaza DOAR pe randuri blocate
|
||||
(gestionabil); sent/sending/queued nu au checkbox (read-only).
|
||||
Butonul "Aplica cod" foloseste hx-post propriu (override form action).
|
||||
hx-disinherit="hx-confirm" pe form => butonul aplica-cod NU mosteneste confirmare. #}
|
||||
<form id="bulk-trimiteri"
|
||||
hx-post="/trimiteri/sterge-bulk"
|
||||
hx-target="#submissions-wrap"
|
||||
@@ -9,31 +36,47 @@
|
||||
hx-disinherit="hx-confirm"
|
||||
style="margin:0;">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<div style="display:flex; justify-content:flex-end; margin-bottom:8px;">
|
||||
<div style="display:flex; justify-content:flex-end; align-items:center;
|
||||
gap:6px; margin-bottom:8px; flex-wrap:wrap;">
|
||||
{# Bulk-fix: input cod + buton aplica (US-010 PRD 5.15) #}
|
||||
<input type="text" name="cod_prestatie" id="bulk-fix-cod"
|
||||
placeholder="Cod RAR (ex: OE-1)"
|
||||
autocomplete="off" autocapitalize="characters"
|
||||
style="width:120px; font-size:12px; padding:3px 7px;
|
||||
border:1px solid var(--line); border-radius:5px;
|
||||
background:var(--card2); color:var(--ink);"
|
||||
aria-label="Cod RAR de aplicat la randurile selectate">
|
||||
<button type="button"
|
||||
hx-post="/trimiteri/bulk-fix"
|
||||
hx-target="#submissions-wrap"
|
||||
hx-swap="innerHTML"
|
||||
style="background:var(--card); color:var(--accent); border-color:var(--accent);
|
||||
font-size:13px; padding:4px 10px; border-radius:5px; cursor:pointer;"
|
||||
aria-label="Aplica codul RAR la randurile blocate selectate">
|
||||
Aplica cod
|
||||
</button>
|
||||
{# Separator vizual #}
|
||||
<span style="color:var(--muted); font-size:11px; padding:0 2px;" aria-hidden="true">|</span>
|
||||
{# Bulk-delete: pastreaza exact comportamentul existent #}
|
||||
<button type="submit" id="bulk-sterge-btn"
|
||||
style="background:var(--card); color:var(--err); border-color:var(--err); font-size:13px; padding:4px 10px;">
|
||||
style="background:var(--card); color:var(--err); border-color:var(--err);
|
||||
font-size:13px; padding:4px 10px; border-radius:5px; cursor:pointer;">
|
||||
Sterge selectate
|
||||
</button>
|
||||
</div>
|
||||
<div class="tablewrap tabel-trimiteri">
|
||||
<table>
|
||||
<thead><tr>
|
||||
<th class="col-chk"><span class="muted" title="Selecteaza randuri blocate">✓</span></th>
|
||||
<th class="col-id">#</th>
|
||||
<th class="col-stare">Stare</th>
|
||||
<th class="col-vehicul">Vehicul</th>
|
||||
<th class="col-operatie">Operatie</th>
|
||||
<th class="col-data">Data prestatie</th>
|
||||
<th class="col-rar">Nr. prezentare RAR</th>
|
||||
<th class="col-actualizat">Actualizat</th>
|
||||
</tr></thead>
|
||||
<tbody>
|
||||
|
||||
{# Lista slim trimiteri (US-004, PRD 5.15).
|
||||
Inlocuieste tabelul cu randuri compacte: VIN mono + operatie·ora + pill.
|
||||
Nr. inmatriculare, data prestatie si nr. prezentare RAR raman accesibile
|
||||
pe linia meta discreta (linia 3) si in modalul de detaliu. #}
|
||||
<ul class="lista-trimiteri-slim" role="list"
|
||||
aria-label="Lista trimiteri">
|
||||
{% for r in rows %}
|
||||
{# PRD 5.9 US-003: randul declanseaza deschiderea MODALULUI global (#detaliu-modal-body),
|
||||
nu un rand-sibling. Clickabil/focusabil (role=button); Enter/Space deschid modalul
|
||||
(JS in base.html). Vechiul rand-sibling de detaliu a fost eliminat. #}
|
||||
<tr id="trimitere-row-{{ r.id }}"
|
||||
class="trimitere-row"
|
||||
{# Randul slim: stanga = VIN mono scurt (L1) + operatie·ora muted (L2) + meta (L3);
|
||||
dreapta = pill de stare. Click deschide modalul global (#detaliu-modal-body).
|
||||
Clickabil/focusabil (role=button); Enter/Space deschid modalul (JS in base.html). #}
|
||||
<li id="trimitere-row-{{ r.id }}"
|
||||
class="trimitere-slim"
|
||||
data-detaliu-id="{{ r.id }}"
|
||||
hx-get="/_fragments/trimitere/{{ r.id }}"
|
||||
hx-target="#detaliu-modal-body"
|
||||
@@ -42,47 +85,163 @@
|
||||
aria-haspopup="dialog"
|
||||
style="cursor:pointer;"
|
||||
title="Click pentru detaliul complet">
|
||||
<td class="col-chk" onclick="event.stopPropagation();">
|
||||
|
||||
{# Zona checkbox — nu declanseaza modalul (stopPropagation).
|
||||
Vizibila DOAR pe randurile gestionabile (error/needs_data/needs_mapping).
|
||||
Latimea fixa previne reflow la prezenta/absenta checkbox-ului. #}
|
||||
<div style="flex:0 0 22px; display:flex; align-items:center;" onclick="event.stopPropagation();">
|
||||
{% if r.gestionabil %}
|
||||
<input type="checkbox" name="submission_id" value="{{ r.id }}"
|
||||
aria-label="Selecteaza trimiterea #{{ r.id }} pentru stergere">
|
||||
{% endif %}
|
||||
</td>
|
||||
<td class="col-id muted" data-eticheta="#">{{ r.id }}</td>
|
||||
<td class="col-stare" data-eticheta="Stare">
|
||||
<span class="pill {{ r.stare_css }}" title="{{ r.stare_text }}">{{ r.stare_scurt }}</span>
|
||||
{# PRD 5.9 US-002 (R1): eticheta umana scurta sub pill — text mic, `s-error`
|
||||
pe error/needs_* (singurele stari pe care `eticheta_problema` e ne-goala).
|
||||
Stare transmisa prin TEXT, nu doar culoare. Codul brut ramane in modal. #}
|
||||
{% if r.eticheta_problema %}
|
||||
</div>
|
||||
|
||||
{# Bloc text principal — stanga, ocupa spatiul ramas. Rand de 2 linii (spec 5.16):
|
||||
L1 = placuta (identificator primar); L2 = cod RAR · operatie · data prestatie. #}
|
||||
<div style="flex:1 1 auto; min-width:0;">
|
||||
|
||||
{# Linia 1: nr. inmatriculare (placuta) — identificatorul primar pe care il
|
||||
scaneaza operatorul. .slim-vin reumplut (acelasi nume de clasa, churn minim).
|
||||
Fallback cand placuta lipseste ('—'): VIN scurt, apoi mesaj neutru
|
||||
(nu randa em-dash izolat ca identificator). #}
|
||||
{% if r.prez.vehicul_nr and r.prez.vehicul_nr != '—' %}
|
||||
<div class="slim-vin">{{ r.prez.vehicul_nr }}</div>
|
||||
{% elif r.prez.vin_scurt and r.prez.vin_scurt != '—' %}
|
||||
<div class="slim-vin muted">{{ r.prez.vin_scurt }}</div>
|
||||
{% else %}
|
||||
<div class="slim-vin muted">fara numar</div>
|
||||
{% endif %}
|
||||
|
||||
{# Linia 2: cod RAR (sau 'nemapat') · operatie (ink, ellipsis) · data prestatie.
|
||||
Separatorul "·" e injectat prin CSS intre celule. Operatia primeste ellipsis
|
||||
ca randul sa NU treaca pe a 3-a linie nici la 390px.
|
||||
VIN integral, #id_prezentare si secundele traiesc in modalul de detaliu. #}
|
||||
<div class="slim-meta slim-rand2">
|
||||
{% if r.prez.cod_rar and r.prez.cod_rar != '—' %}
|
||||
<span class="cod-rar-cod">{{ r.prez.cod_rar }}</span>
|
||||
{% else %}
|
||||
<span class="cod-rar-cod cod-rar-sub muted">nemapat</span>
|
||||
{% endif %}
|
||||
<span class="slim-op">{{ r.prez.operatie }}</span>
|
||||
{% if r.prez.data_prestatie and r.prez.data_prestatie != '—' %}
|
||||
<span class="slim-data muted">{{ r.prez.data_prestatie }}</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
{# Micro-linie umana a problemei — text mic s-error, DOAR pe stari de problema
|
||||
(loud-on-exception D6). Randul normal/finalizat ramane strict 2 linii.
|
||||
Token tipografic --fs-xs (>=12px, scala 5.16). #}
|
||||
{% if r.eticheta_problema and r.eticheta_problema != r.stare_scurt and r.eticheta_problema != r.stare_text %}
|
||||
<div class="eticheta-problema s-error">{{ r.eticheta_problema }}</div>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td class="col-vehicul" data-eticheta="Vehicul">
|
||||
{{ r.prez.vehicul_nr }}
|
||||
{% if r.prez.vin_scurt and r.prez.vin_scurt != '—' %}
|
||||
<span class="muted" style="font-size:12px;">{{ r.prez.vin_scurt }}</span>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td class="col-operatie" data-eticheta="Operatie">
|
||||
<div>{{ r.prez.operatie }}</div>
|
||||
{# PRD 5.9 US-002: doar codul RAR (ex. OE-2), FARA prefixul "cod RAR:" — chip
|
||||
muted discret; cand nemapat afiseaza "nemapat" muted (comportament 5.8). #}
|
||||
{% if r.prez.cod_rar and r.prez.cod_rar != '—' %}
|
||||
<div class="cod-rar-sub"><span class="cod-rar-cod">{{ r.prez.cod_rar }}</span></div>
|
||||
{% else %}
|
||||
<div class="muted cod-rar-sub">nemapat</div>
|
||||
{% endif %}
|
||||
</td>
|
||||
<td class="col-data" data-eticheta="Data prestatie">{{ r.prez.data_prestatie }}</td>
|
||||
<td class="col-rar" data-eticheta="Nr. prezentare RAR">{{ r.id_prezentare or '—' }}</td>
|
||||
<td class="col-actualizat muted" data-eticheta="Actualizat">{{ r.updated_at }}</td>
|
||||
</tr>
|
||||
</div>
|
||||
|
||||
{# Pill de stare — dreapta, flex:none #}
|
||||
<span class="pill {{ r.stare_css }}" title="{{ r.stare_text }}"
|
||||
style="flex:0 0 auto; white-space:nowrap;">{{ r.stare_scurt }}</span>
|
||||
</li>
|
||||
{% endfor %}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</ul>
|
||||
</form>
|
||||
|
||||
{#
|
||||
Paginare numerotata.
|
||||
Afisata doar cand exista mai mult de o pagina.
|
||||
Fiecare link pastreaza filtrele curente (status, vehicul, data_de, data_pana).
|
||||
Pagina curenta: aria-current="page" (semantic).
|
||||
#}
|
||||
{% if total is defined %}
|
||||
<div aria-live="polite"
|
||||
style="font-size:12px; color:var(--muted); text-align:right; margin-top:6px; margin-bottom:2px;">
|
||||
{% if total == 0 %}
|
||||
0 trimiteri
|
||||
{% else %}
|
||||
{{ page_start }}–{{ page_end }} din {{ total }}
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{% if pages is defined and pages > 1 %}
|
||||
{#
|
||||
Construim param-string pentru filtrele curente (fara page) — refolosit in fiecare link.
|
||||
Filtrul status vine din pill-uri (nu din form); il pastram in URL.
|
||||
#}
|
||||
{% set pq = "" %}
|
||||
{% if f_status %}{% set pq = pq + "&status=" + f_status %}{% endif %}
|
||||
{% if f_vehicul %}{% set pq = pq + "&vehicul=" + f_vehicul %}{% endif %}
|
||||
{% if f_data_de %}{% set pq = pq + "&data_de=" + f_data_de %}{% endif %}
|
||||
{% if f_data_pana %}{% set pq = pq + "&data_pana=" + f_data_pana %}{% endif %}
|
||||
|
||||
<nav aria-label="Paginare trimiteri"
|
||||
style="display:flex; justify-content:center; gap:4px; flex-wrap:wrap; margin-top:10px;">
|
||||
|
||||
{# Buton Anterior #}
|
||||
{% if page > 1 %}
|
||||
<button type="button"
|
||||
hx-get="/_fragments/submissions?page={{ page - 1 }}{{ pq }}"
|
||||
hx-target="#submissions-wrap"
|
||||
hx-swap="innerHTML"
|
||||
style="padding:3px 10px; border-radius:6px; font-size:13px; cursor:pointer;
|
||||
border:1px solid var(--line); background:var(--card); color:var(--fg);"
|
||||
aria-label="Pagina anterioara">
|
||||
«
|
||||
</button>
|
||||
{% else %}
|
||||
<button type="button" disabled
|
||||
style="padding:3px 10px; border-radius:6px; font-size:13px;
|
||||
border:1px solid var(--line); background:var(--card); color:var(--muted);
|
||||
opacity:0.4; cursor:default;"
|
||||
aria-label="Pagina anterioara (indisponibila)">
|
||||
«
|
||||
</button>
|
||||
{% endif %}
|
||||
|
||||
{# Numerele de pagina #}
|
||||
{% for p in range(1, pages + 1) %}
|
||||
{% if p == page %}
|
||||
<button type="button"
|
||||
aria-current="page"
|
||||
style="padding:3px 10px; border-radius:6px; font-size:13px; cursor:default;
|
||||
border:1px solid var(--accent); background:var(--accent); color:#fff;
|
||||
font-weight:700;">
|
||||
{{ p }}
|
||||
</button>
|
||||
{% else %}
|
||||
<button type="button"
|
||||
hx-get="/_fragments/submissions?page={{ p }}{{ pq }}"
|
||||
hx-target="#submissions-wrap"
|
||||
hx-swap="innerHTML"
|
||||
style="padding:3px 10px; border-radius:6px; font-size:13px; cursor:pointer;
|
||||
border:1px solid var(--line); background:var(--card); color:var(--fg);">
|
||||
{{ p }}
|
||||
</button>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
|
||||
{# Buton Urmator #}
|
||||
{% if page < pages %}
|
||||
<button type="button"
|
||||
hx-get="/_fragments/submissions?page={{ page + 1 }}{{ pq }}"
|
||||
hx-target="#submissions-wrap"
|
||||
hx-swap="innerHTML"
|
||||
style="padding:3px 10px; border-radius:6px; font-size:13px; cursor:pointer;
|
||||
border:1px solid var(--line); background:var(--card); color:var(--fg);"
|
||||
aria-label="Pagina urmatoare">
|
||||
»
|
||||
</button>
|
||||
{% else %}
|
||||
<button type="button" disabled
|
||||
style="padding:3px 10px; border-radius:6px; font-size:13px;
|
||||
border:1px solid var(--line); background:var(--card); color:var(--muted);
|
||||
opacity:0.4; cursor:default;"
|
||||
aria-label="Pagina urmatoare (indisponibila)">
|
||||
»
|
||||
</button>
|
||||
{% endif %}
|
||||
|
||||
</nav>
|
||||
{% endif %}
|
||||
|
||||
{% elif filtru_activ %}
|
||||
<div class="empty">
|
||||
Nimic pe filtrul curent.
|
||||
|
||||
@@ -1,14 +1,13 @@
|
||||
{% from "_eroare.html" import card_erori %}
|
||||
{% import '_macros.html' as ui %}
|
||||
{# PRD 5.9 US-004: detaliu editabil in-place, butoane consolidate, ordine verticala R10.
|
||||
Fragmentul se swap-uieste in corpul modalului global (#detaliu-modal-body). Heading-ul
|
||||
poarta id-ul folosit de aria-labelledby al dialogului.
|
||||
R9: operatie + cod RAR rezolvat apar IMPREUNA, read-only, folosind `prez.cod_rar`
|
||||
{# Detaliu editabil in-place. Fragmentul se swap-uieste in corpul modalului global
|
||||
(#detaliu-modal-body). Heading-ul poarta id-ul folosit de aria-labelledby al dialogului.
|
||||
Operatie + cod RAR rezolvat apar IMPREUNA, read-only, folosind `prez.cod_rar`
|
||||
(fallback „nemapat"), fara eticheta separata „Cod RAR". #}
|
||||
{% set cod_afis = prez.cod_rar if (prez.cod_rar and prez.cod_rar != '—') else 'nemapat' %}
|
||||
<div class="card" id="detaliu-card-{{ id }}" style="border:none; padding:0; margin:0;">
|
||||
|
||||
{# === R10 (1): header — #id + pill + motiv uman === #}
|
||||
{# === Header — #id + pill + motiv uman === #}
|
||||
<div style="display:flex; align-items:center; gap:8px; flex-wrap:wrap; margin:0 0 8px;">
|
||||
<h2 id="detaliu-modal-titlu" style="font-size:15px; margin:0;">Detaliu trimitere #{{ id }}</h2>
|
||||
<span class="pill {{ stare_css }}">{{ stare_text }}</span>
|
||||
@@ -19,14 +18,16 @@
|
||||
<p class="muted" style="margin:0 0 12px; font-size:13px;">{{ stare_subtext }}</p>
|
||||
{% endif %}
|
||||
|
||||
{# === R10 (2): bloc eroare blocanta cand exista === #}
|
||||
{% if erori_3n %}
|
||||
{# === Bloc eroare blocanta — DOAR in read-only.
|
||||
In editare, cardul 3-niveluri e inlocuit cu: erori per-camp in macro `camp`
|
||||
(text simplu .s-error) + rezumat top-of-form pentru erori fara camp (mai jos). === #}
|
||||
{% if not editabil and erori_3n %}
|
||||
<div style="margin:0 0 14px;">
|
||||
{{ card_erori(erori_3n) }}
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === R10 (3) + R9: mapare inline (PRD 5.7) — alege cod RAR pentru operatiile nemapate.
|
||||
{# === Mapare inline — alege cod RAR pentru operatiile nemapate.
|
||||
Cand nemapate_inline, linia „Operatie: X · nemapat" apare in formularul de mai jos
|
||||
(cod_afis = nemapat), iar aici e picker-ul; dupa mapare, re-render arata codul rezolvat. === #}
|
||||
{% if nemapate_inline %}
|
||||
@@ -68,7 +69,6 @@
|
||||
</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
{{ ui.autosend_toggle(checked=True) }}
|
||||
<button type="submit">Salveaza maparea</button>
|
||||
</div>
|
||||
</form>
|
||||
@@ -76,57 +76,53 @@
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === R10 (4): formular editabil (needs_data/needs_mapping) SAU context read-only.
|
||||
{# === Formular editabil (needs_data/needs_mapping) SAU context read-only.
|
||||
Zero dublare: campurile vehiculului apar O SINGURA DATA — editabile cand randul e
|
||||
corectabil, altfel read-only. Operatie + cod RAR read-only deasupra campurilor. === #}
|
||||
{% if editabil %}
|
||||
{% set err_map = {} %}
|
||||
{% for e in corectie_errors %}{% if e.field %}{% set _ = err_map.update({e.field: e.message}) %}{% endif %}{% endfor %}
|
||||
{# fix_map gol pentru Trimiteri (fix-hints vin din preview, nu din corectii de trimitere). #}
|
||||
{% set fix_map = {} %}
|
||||
{# vin_context pentru aria-label cu context VIN (D#6). #}
|
||||
{%- set vin_context = form_vin -%}
|
||||
{# btn_label pentru butonul primar al partial-ului. #}
|
||||
{%- set btn_label = 'Salveaza si retrimite' -%}
|
||||
|
||||
{% if corectie_msg %}
|
||||
<div class="flash" style="{% if corectie_error %}border-color:var(--err); background:color-mix(in srgb, var(--err) 12%, var(--card));{% endif %} margin:0 0 12px;"
|
||||
{% if corectie_error %}role="alert"{% endif %}>{{ corectie_msg }}</div>
|
||||
{% endif %}
|
||||
|
||||
{% macro camp(nume, eticheta, valoare, tip='text') %}
|
||||
<div style="margin-bottom:10px;">
|
||||
<label for="c-{{ nume }}" class="muted" style="font-size:12px; display:block;">{{ eticheta }}</label>
|
||||
<input id="c-{{ nume }}" type="{{ tip }}" name="{{ nume }}" value="{{ valoare }}"
|
||||
style="width:100%; {% if err_map.get(nume) %}border-color:var(--err);{% endif %}"
|
||||
{% if err_map.get(nume) %}aria-invalid="true"{% endif %}>
|
||||
{% if err_map.get(nume) %}
|
||||
<div class="s-error" style="font-size:12px; margin-top:2px;">{{ err_map.get(nume) }}</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endmacro %}
|
||||
{# Erori fara camp (field None) nu dispar silentios in editare —
|
||||
cardul 3n e ascuns, deci adaugam un rezumat simplu top-of-form.
|
||||
Erori cu camp raman afisate per-camp de macro-ul `camp` din _form_editare.html. #}
|
||||
{% for e in erori_3n if not e.field %}
|
||||
<div class="s-error" style="font-size:13px; margin:0 0 10px;" role="alert">{{ e.problema }}</div>
|
||||
{% endfor %}
|
||||
|
||||
<form hx-post="/trimitere/{{ id }}/corecteaza"
|
||||
hx-target="#detaliu-modal-body" hx-swap="innerHTML"
|
||||
hx-disabled-elt="find button">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
|
||||
{# Operatie + cod RAR read-only deasupra campurilor (R9, fara eticheta „Cod RAR"). #}
|
||||
{# Cleanup B (US-009 PRD 5.15): vechiul <select name="cod_prestatie"> eliminat.
|
||||
Chips din _form_editare.html (via _chips_prestatii.html) il inlocuiesc complet:
|
||||
emit hidden inputs name="cod_prestatie" + picker per-operatie (E4, US-007).
|
||||
post_corectie_trimitere foloseste form.getlist("cod_prestatie") → compatibil. #}
|
||||
|
||||
{# Operatie service (cod intern + denumire venita prin API/import), distinct de
|
||||
operatia RAR mapata. op_service_cod="" cand lipseste → randul absent.
|
||||
RAMANE in _trimitere_detaliu.html (D#5). #}
|
||||
{% if prez.op_service_cod %}
|
||||
<div style="margin:0 0 12px;">
|
||||
<div class="muted" style="font-size:12px;">Operatie</div>
|
||||
<div>{{ prez.operatie }} · {{ cod_afis }}</div>
|
||||
<div class="muted" style="font-size:12px;">Operatie service</div>
|
||||
<div>{{ prez.op_service_cod }}{% if prez.op_service_denumire %} — {{ prez.op_service_denumire }}{% endif %}</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# Nr. inmatriculare pe rand propriu, VIN dedesubt — ambele latime plina. #}
|
||||
{{ camp('nr_inmatriculare', 'Numar inmatriculare', form_nr) }}
|
||||
{{ camp('vin', 'VIN (serie sasiu)', form_vin) }}
|
||||
|
||||
{# Restul campurilor in grila. #}
|
||||
<div style="display:grid; grid-template-columns:repeat(auto-fit, minmax(200px, 1fr)); gap:0 16px;">
|
||||
{{ camp('data_prestatie', 'Data prestatie (YYYY-MM-DD)', form_data) }}
|
||||
{{ camp('odometru_final', 'Odometru final', form_odo_final) }}
|
||||
{{ camp('odometru_initial', 'Odometru initial (daca e cerut)', form_odo_initial) }}
|
||||
</div>
|
||||
|
||||
{# === R10 (5): actiune primara conditionata de stare (R2). needs_data/needs_mapping
|
||||
-> „Salveaza si retrimite" pe /corecteaza. UN SINGUR buton primar per stare. === #}
|
||||
<div style="margin-top:14px;">
|
||||
<button type="submit">Salveaza si retrimite</button>
|
||||
</div>
|
||||
{# === Campurile vehicul/data/odo + erori/fix + buton — partial DRY (US-005). === #}
|
||||
{% include "_form_editare.html" %}
|
||||
</form>
|
||||
{% else %}
|
||||
{# Context read-only pentru randuri ne-editabile (sent/sending/queued/error). #}
|
||||
@@ -139,25 +135,49 @@
|
||||
<div style="word-break:break-all;">{{ prez.vin }}</div>
|
||||
</div>
|
||||
<div><div class="muted" style="font-size:12px;">Operatie</div><div>{{ prez.operatie }} · {{ cod_afis }}</div></div>
|
||||
{# Operatie service (cod intern + denumire), distinct de operatia RAR.
|
||||
op_service_cod="" cand lipseste → randul absent (fara "—"). #}
|
||||
{% if prez.op_service_cod %}
|
||||
<div><div class="muted" style="font-size:12px;">Operatie service</div>
|
||||
<div>{{ prez.op_service_cod }}{% if prez.op_service_denumire %} — {{ prez.op_service_denumire }}{% endif %}</div></div>
|
||||
{% endif %}
|
||||
<div><div class="muted" style="font-size:12px;">Data prestatie</div><div>{{ prez.data_prestatie }}</div></div>
|
||||
<div><div class="muted" style="font-size:12px;">Odometru final</div><div>{{ prez.odometru }}</div></div>
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === R10 (5): actiuni de jos — primar Re-pune (doar error) + Sterge pe RAND SEPARAT (R2/R11) === #}
|
||||
{# === Actiuni de jos — primar Re-pune (doar error) + Sterge pe RAND SEPARAT === #}
|
||||
{% if status == 'error' or gestionabil %}
|
||||
<div class="detaliu-actiuni-jos" style="margin-top:14px; padding-top:12px; border-top:1px solid var(--line);">
|
||||
{# R2: error -> buton primar „Re-pune in coada" pe /repune (error nu e editabil). #}
|
||||
{# Error -> buton primar „Re-pune in coada" pe /repune (error nu e editabil pentru #}
|
||||
{# campuri vehicul, dar se poate schimba cod_prestatie prin acelasi formular). #}
|
||||
{% if status == 'error' %}
|
||||
<form hx-post="/trimitere/{{ id }}/repune"
|
||||
hx-target="#detaliu-modal-body" hx-swap="innerHTML"
|
||||
hx-disabled-elt="find button" style="margin:0 0 10px;">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
{# Select cod_prestatie optional in formularul /repune (doar pentru error). #}
|
||||
{% if nomenclator_rar %}
|
||||
<label for="cod-rar-error-{{ id }}" style="display:block; font-size:12px; color:var(--muted); margin-bottom:4px;">
|
||||
Operatie RAR (optional — schimba codul si re-pune)
|
||||
</label>
|
||||
<select id="cod-rar-error-{{ id }}" name="cod_prestatie"
|
||||
aria-label="Alege operatia RAR din nomenclator"
|
||||
style="width:100%; margin-bottom:8px; font-size:13px;">
|
||||
<option value="">— pastrat ({{ cod_prestatie_curent }}) —</option>
|
||||
{% for item in nomenclator_rar %}
|
||||
<option value="{{ item.cod_prestatie }}"
|
||||
{% if item.cod_prestatie == cod_prestatie_curent %}selected{% endif %}>
|
||||
{{ item.cod_prestatie }} — {{ item.nume_prestatie }}
|
||||
</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
{% endif %}
|
||||
<button type="submit">Re-pune in coada</button>
|
||||
</form>
|
||||
{% endif %}
|
||||
|
||||
{# R11: UN SINGUR Sterge, outline distructiv (var(--err)), pe rand separat, full-width pe mobil. #}
|
||||
{# UN SINGUR Sterge, outline distructiv (var(--err)), pe rand separat, full-width pe mobil. #}
|
||||
{% if gestionabil %}
|
||||
<form hx-post="/trimitere/{{ id }}/sterge"
|
||||
hx-target="#detaliu-modal-body" hx-swap="innerHTML"
|
||||
@@ -173,7 +193,7 @@
|
||||
</div>
|
||||
{% endif %}
|
||||
|
||||
{# === R10 (6): Detalii tehnice — colapsat implicit === #}
|
||||
{# === Detalii tehnice — colapsat implicit === #}
|
||||
<details style="margin-top:14px;">
|
||||
<summary class="muted" style="font-size:12px; cursor:pointer;">Detalii tehnice</summary>
|
||||
<div style="display:grid; grid-template-columns:repeat(auto-fit, minmax(200px, 1fr)); gap:10px 24px; margin-top:10px;">
|
||||
@@ -195,7 +215,6 @@
|
||||
{% endif %}
|
||||
</details>
|
||||
</div>
|
||||
{# PRD 5.9 US-004 (R4): scriptul inline vechi (marcheazaDetaliuDeschis / scrollIntoView pe
|
||||
randul-sibling) a fost eliminat de US-003. Focus-ul post-swap (incl. re-render corectie/
|
||||
mapare) e gestionat de htmx:afterSettle pe #detaliu-modal-body din base.html. R5: inchiderea
|
||||
modalului pe succes (queued/sterge) vine din HX-Trigger `inchideModal` emis de rute. #}
|
||||
{# Focus-ul post-swap (incl. re-render corectie/mapare) e gestionat de htmx:afterSettle pe
|
||||
#detaliu-modal-body din base.html. Inchiderea modalului pe succes (queued/sterge) vine
|
||||
din HX-Trigger `inchideModal` emis de rute. #}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<div id="import-section">
|
||||
{% set pas = 1 %}{% include '_stepper.html' %}
|
||||
{# US-004 (3.6): bara de upload accentuata (border de accent) ca sa ramana punctul
|
||||
de intrare evident chiar cu tabelul Trimiteri lung dedesubt (D-1.1/D-5.2). #}
|
||||
{# Bara de upload accentuata (border de accent) ca sa ramana punctul
|
||||
de intrare evident chiar cu tabelul Trimiteri lung dedesubt. #}
|
||||
{% from '_eroare.html' import card_erori %}
|
||||
<div class="card" style="border-color:var(--accent);">
|
||||
|
||||
@@ -52,15 +52,15 @@
|
||||
role="region" aria-label="Zona de incarcare fisier"
|
||||
style="display:flex; align-items:center; gap:14px; flex-wrap:wrap;
|
||||
padding:12px 16px; text-align:left;">
|
||||
<strong style="font-size:14px;">Importa:</strong>
|
||||
<strong style="font-size:var(--fs-md);">Importa:</strong>
|
||||
<input id="file-input" type="file" name="file" accept=".xlsx,.xls,.csv"
|
||||
style="display:none;" aria-label="Selecteaza fisier xlsx sau csv">
|
||||
<button type="button" id="upload-btn"
|
||||
style="min-height:44px; padding:10px 20px; font-size:14px;">
|
||||
style="min-height:44px; padding:10px 20px; font-size:var(--fs-md);">
|
||||
Alege fisier (xlsx/csv)
|
||||
</button>
|
||||
<span class="muted" style="font-size:13px;">sau trage aici</span>
|
||||
<span class="muted" style="font-size:12px; margin-left:auto;">
|
||||
<span class="muted" style="font-size:var(--fs-sm);">sau trage aici</span>
|
||||
<span class="muted" style="font-size:var(--fs-xs); margin-left:auto;">
|
||||
NU se trimite nimic la RAR pana confirmi.
|
||||
</span>
|
||||
</div>
|
||||
@@ -69,10 +69,10 @@
|
||||
<div class="drop-zone" id="drop-zone"
|
||||
role="region" aria-label="Zona de incarcare fisier">
|
||||
{% if not sheets %}
|
||||
<p style="font-size:17px; margin:0 0 4px; font-weight:600;">Primul fisier? Trage-l aici.</p>
|
||||
<p class="muted" style="margin:0 0 16px; font-size:13px;">xlsx sau csv, max 5000 randuri</p>
|
||||
<p style="font-size:var(--fs-lg); margin:0 0 4px; font-weight:600;">Primul fisier? Trage-l aici.</p>
|
||||
<p class="muted" style="margin:0 0 16px; font-size:var(--fs-sm);">xlsx sau csv, max 5000 randuri</p>
|
||||
{% else %}
|
||||
<p class="muted" style="margin:0 0 16px; font-size:14px;">
|
||||
<p class="muted" style="margin:0 0 16px; font-size:var(--fs-md);">
|
||||
Incarca fisierul din nou dupa ce ai ales foaia.
|
||||
</p>
|
||||
{% endif %}
|
||||
@@ -80,18 +80,18 @@
|
||||
<input id="file-input" type="file" name="file" accept=".xlsx,.xls,.csv"
|
||||
style="display:none;" aria-label="Selecteaza fisier xlsx sau csv">
|
||||
<button type="button" id="upload-btn"
|
||||
style="min-height:44px; padding:10px 24px; font-size:14px;">
|
||||
style="min-height:44px; padding:10px 24px; font-size:var(--fs-md);">
|
||||
Alege fisier (xlsx/csv)
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<p class="muted" style="margin:8px 0 0; font-size:12px;">
|
||||
<p class="muted" style="margin:8px 0 0; font-size:var(--fs-xs);">
|
||||
NU se trimite nimic la RAR pana confirmi explicit.
|
||||
</p>
|
||||
{% endif %}
|
||||
|
||||
<span id="upload-spinner" class="htmx-indicator muted"
|
||||
style="font-size:13px; margin-top:6px; display:inline;">
|
||||
style="font-size:var(--fs-sm); margin-top:6px; display:inline;">
|
||||
se parseaza fisierul...
|
||||
</span>
|
||||
</form>
|
||||
@@ -105,7 +105,7 @@
|
||||
var dz = document.getElementById('drop-zone');
|
||||
var frm = document.getElementById('upload-form');
|
||||
|
||||
/* US-003 (3.6): un singur sticky bar pe ecran — cand re-apare zona de upload
|
||||
/* Un singur sticky bar pe ecran — cand re-apare zona de upload
|
||||
(reset sau dupa commit), sectiunea Trimiteri redevine vizibila. */
|
||||
var trim = document.getElementById('trimiteri-section');
|
||||
if (trim) trim.style.display = '';
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
{% block title %}Conturi clienti — Gateway RAR AUTOPASS{% endblock %}
|
||||
{% block content %}
|
||||
|
||||
{# US-009 (5.5): metadate verbe de ciclu de viata (eticheta, ruta, clasa). #}
|
||||
{# Metadate verbe de ciclu de viata (eticheta, ruta, clasa). #}
|
||||
{% set VERBS = {
|
||||
'activate': ('Activeaza', '/admin/activate', ''),
|
||||
'block': ('Blocheaza', '/admin/block', ''),
|
||||
@@ -10,6 +10,9 @@
|
||||
'delete': ('Sterge', '/admin/delete', 'danger')
|
||||
} %}
|
||||
|
||||
{# Tier-uri selectabile in panou (cod, eticheta). Aliniat cu app/plans.py#PLANS. #}
|
||||
{% set TIERS = [('free', 'Gratuit'), ('standard', 'Standard'), ('pro', 'Pro'), ('premium', 'Premium')] %}
|
||||
|
||||
{% macro lifecycle_block(title, rows, block_id, bulk_verbs, row_verbs) %}
|
||||
<div class="card">
|
||||
<h3 style="margin-top:0;">{{ title }} ({{ rows|length }})</h3>
|
||||
@@ -34,7 +37,7 @@
|
||||
<thead><tr>
|
||||
<th style="width:28px;"><input type="checkbox" class="master-check" data-block="{{ block_id }}"
|
||||
aria-label="Selecteaza tot"></th>
|
||||
<th>ID</th><th>Companie</th><th>CUI</th><th>Email</th><th>Stare</th><th>Inregistrat</th><th>Actiuni</th>
|
||||
<th>ID</th><th>Companie</th><th>CUI</th><th>Email</th><th>Plan curent</th><th>Plan cerut</th><th>Stare</th><th>Inregistrat</th><th>Actiuni</th>
|
||||
</tr></thead>
|
||||
<tbody>
|
||||
{% for acct in rows %}
|
||||
@@ -46,6 +49,45 @@
|
||||
<td>{{ acct.name }}</td>
|
||||
<td class="muted">{{ acct.cui or "—" }}</td>
|
||||
<td>{{ acct.email or "—" }}</td>
|
||||
<td style="white-space:nowrap;">
|
||||
{# Plan EFECTIV acum (prominent): trial Pro activ ridica free->pro. #}
|
||||
<div style="margin-bottom:5px;">
|
||||
<span class="pill" style="font-weight:600;">{{ acct.tier_efectiv_label }}</span>
|
||||
{% if acct.trial_activ %}
|
||||
<span class="muted" style="font-size:11px;">
|
||||
trial{% if acct.trial_zile %} · {{ acct.trial_zile }} {{ 'zi' if acct.trial_zile == 1 else 'zile' }} ramase{% endif %}
|
||||
→ apoi {{ acct.tier_label }}
|
||||
</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
{# Schimbare plan inline: select tier de baza + Aplica. Form propriu (nu imbricat in bulk-form).
|
||||
Aplica INCHEIE trial-ul si seteaza planul ales ca real, cu efect imediat. #}
|
||||
<form method="post" action="/admin/set-tier" class="tier-form"
|
||||
style="display:flex;align-items:center;gap:6px;">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<input type="hidden" name="account_id" value="{{ acct.id }}">
|
||||
<select name="tier" aria-label="Plan pentru {{ acct.name }}"
|
||||
style="padding:4px 8px;min-height:32px;max-width:130px;">
|
||||
{% for code, label in TIERS %}
|
||||
<option value="{{ code }}"{% if acct.tier == code %} selected{% endif %}>{{ label }}</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
<button type="submit" class="btn-sm"
|
||||
title="Aplica planul ales ca plan real (incheie trial-ul daca e activ)">Aplica</button>
|
||||
</form>
|
||||
{# Acorda/prelungeste trial Pro de N zile, fara a schimba tier-ul de baza. #}
|
||||
<form method="post" action="/admin/set-trial" class="trial-form"
|
||||
style="display:flex;align-items:center;gap:6px;margin-top:5px;">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<input type="hidden" name="account_id" value="{{ acct.id }}">
|
||||
<input type="number" name="trial_days" value="30" min="1" max="3650"
|
||||
aria-label="Zile trial Pro pentru {{ acct.name }}"
|
||||
style="padding:4px 8px;min-height:32px;width:64px;">
|
||||
<button type="submit" class="btn-sm"
|
||||
title="Acorda/prelungeste trial Pro de la acum (nu schimba tier-ul de baza)">Trial Pro</button>
|
||||
</form>
|
||||
</td>
|
||||
<td class="muted">{{ acct.requested_plan_label }}</td>
|
||||
<td><span class="pill">{{ acct.status }}</span></td>
|
||||
<td class="muted">{{ acct.created_at or "—" }}</td>
|
||||
<td style="white-space:nowrap;">
|
||||
@@ -60,7 +102,13 @@
|
||||
{% if v == 'delete' %}onsubmit="return confirm('Stergi acest cont? (stergere soft)');"{% endif %}>
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<input type="hidden" name="account_id" value="{{ acct.id }}">
|
||||
{% if v == 'activate' and not acct.is_complete %}
|
||||
<button type="submit"{% if cls == 'danger' %} class="danger"{% endif %}
|
||||
disabled
|
||||
title="Completeaza datele firmei (companie + email + CUI) inainte de activare">{{ label }}</button>
|
||||
{% else %}
|
||||
<button type="submit"{% if cls == 'danger' %} class="danger"{% endif %}>{{ label }}</button>
|
||||
{% endif %}
|
||||
</form>
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -1,91 +1,17 @@
|
||||
{% extends "base.html" %}
|
||||
{% block content %}
|
||||
|
||||
{# US-007 (5.5): nav-ul ad-hoc (Panou admin + logout) a fost mutat in meniul de cont (☰)
|
||||
din header (base.html). Aici raman doar bara de status + tab-bar-ul de lucru zilnic. #}
|
||||
|
||||
<!-- Bara de status (US-002): mereu vizibila, deasupra tab-bar-ului -->
|
||||
<!-- Bara de status: mereu vizibila -->
|
||||
<div id="status-bar" class="status-bar card"
|
||||
hx-get="/_fragments/status"
|
||||
hx-trigger="load, every 15s"
|
||||
hx-get="/_fragments/status?tab={{ active_tab }}"
|
||||
hx-trigger="load, every 15s, trimiteriChanged from:body"
|
||||
hx-swap="outerHTML">
|
||||
<div class="empty muted" style="padding:8px 0;">se incarca starea…</div>
|
||||
</div>
|
||||
|
||||
<!-- Tab-bar: navigare intre sectiuni -->
|
||||
<div role="tablist" class="tab-bar" aria-label="Sectiuni dashboard">
|
||||
{# US-007 (5.5): tab-bar redus la suprafetele de LUCRU ZILNIC (Acasa·Mapari).
|
||||
Cont/Integrare/Nomenclator s-au mutat in meniul de cont (☰) din header — rutele
|
||||
`/_fragments/{cont,integrare,nomenclator}` + deep-link `?tab=` raman valide. #}
|
||||
{% set tabs = [
|
||||
("acasa", "Acasa", "tab-acasa"),
|
||||
("mapari", "Mapari", "tab-mapari")
|
||||
] %}
|
||||
{% for tab_id, tab_label, tab_elem_id in tabs %}
|
||||
{% set badge = (badges.get(tab_id, 0) if badges else 0) %}
|
||||
<a id="{{ tab_elem_id }}"
|
||||
role="tab"
|
||||
href="/?tab={{ tab_id }}"
|
||||
aria-selected="{{ 'true' if active_tab == tab_id else 'false' }}"
|
||||
aria-controls="tab-panel"
|
||||
{% if badge %}aria-label="{{ tab_label }}, {{ badge }} necesita atentie"{% endif %}
|
||||
class="tab-link{% if active_tab == tab_id %} tab-activ{% endif %}"
|
||||
tabindex="{{ '0' if active_tab == tab_id else '-1' }}"
|
||||
hx-get="/_fragments/{{ tab_id }}"
|
||||
hx-target="#tab-panel"
|
||||
hx-swap="innerHTML"
|
||||
hx-push-url="/?tab={{ tab_id }}">{{ tab_label }}{% if badge %}<span class="tab-badge" aria-hidden="true" style="display:inline-flex; align-items:center; justify-content:center; min-width:18px; height:18px; margin-left:6px; padding:0 5px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;">{{ badge }}</span>{% endif %}</a>
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
||||
<!-- Panou activ: randat server-side la full load; HTMX inlocuieste continutul la click pe tab -->
|
||||
<div id="tab-panel" role="tabpanel" aria-labelledby="tab-{{ active_tab }}" class="tab-panel">
|
||||
<!-- Panou activ: randat server-side la full load (Acasa implicit, sau ?tab= prin meniu) -->
|
||||
<div id="tab-panel" class="tab-panel">
|
||||
{{ panel_html | safe }}
|
||||
</div>
|
||||
|
||||
<script>
|
||||
(function() {
|
||||
/* Navigare cu sageti intre tab-uri (ARIA pattern) — scoped pe fiecare tablist.
|
||||
Folosim querySelectorAll pentru a suporta multiple tablist-uri pe pagina
|
||||
(tab-bar principal + tab-urile interne din panoul Integrare). */
|
||||
document.querySelectorAll('[role="tablist"]').forEach(function(tablist) {
|
||||
var tabs = Array.from(tablist.querySelectorAll('[role="tab"]'));
|
||||
if (!tabs.length) return;
|
||||
|
||||
tablist.addEventListener('keydown', function(e) {
|
||||
var idx = tabs.indexOf(document.activeElement);
|
||||
if (idx === -1) return;
|
||||
var next = -1;
|
||||
if (e.key === 'ArrowRight' || e.key === 'ArrowDown') {
|
||||
next = (idx + 1) % tabs.length;
|
||||
} else if (e.key === 'ArrowLeft' || e.key === 'ArrowUp') {
|
||||
next = (idx - 1 + tabs.length) % tabs.length;
|
||||
} else if (e.key === 'Home') {
|
||||
next = 0;
|
||||
} else if (e.key === 'End') {
|
||||
next = tabs.length - 1;
|
||||
}
|
||||
if (next !== -1) {
|
||||
e.preventDefault();
|
||||
tabs[next].focus();
|
||||
}
|
||||
});
|
||||
|
||||
/* La click pe tab: actualizeaza aria-selected + tabindex (scoped pe tablist-ul curent) */
|
||||
tabs.forEach(function(tab) {
|
||||
tab.addEventListener('click', function() {
|
||||
tabs.forEach(function(t) {
|
||||
t.setAttribute('aria-selected', 'false');
|
||||
t.setAttribute('tabindex', '-1');
|
||||
t.classList.remove('tab-activ');
|
||||
});
|
||||
tab.setAttribute('aria-selected', 'true');
|
||||
tab.setAttribute('tabindex', '0');
|
||||
tab.classList.add('tab-activ');
|
||||
});
|
||||
});
|
||||
});
|
||||
})();
|
||||
</script>
|
||||
|
||||
{% endblock %}
|
||||
|
||||
464
app/web/templates/landing.html
Normal file
464
app/web/templates/landing.html
Normal file
@@ -0,0 +1,464 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>ROA AUTOPASS — declari prestațiile la RAR din câteva click-uri</title>
|
||||
<meta name="description" content="Încarci fișierul tău cu operațiile service-ului, completezi o dată codurile RAR și le salvezi. ROMFAST trimite prestațiile la RAR AUTOPASS în locul tău, fără tastat manual. Conform Legii 142/2023.">
|
||||
<style>
|
||||
|
||||
/* US-001/US-008 (PRD 5.16): IBM Plex eliminat complet — stive font sistem standard web.
|
||||
Tokenurile --font-ui / --font-mono definite in :root (sursa unica de adevar). */
|
||||
:root{--font-ui:system-ui,-apple-system,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif;--font-mono:ui-monospace,"SF Mono","Cascadia Code","Segoe UI Mono","Roboto Mono",Menlo,Consolas,monospace;}
|
||||
*{box-sizing:border-box;}
|
||||
html,body{margin:0;padding:0;}
|
||||
body{font-family:var(--font-ui);-webkit-font-smoothing:antialiased;background:var(--bg,#0f1218);color:var(--text,#e6e9ef);}
|
||||
body[data-theme="grafit"]{--bg:#0f1218;--card:#181c24;--card2:#0f1218;--text:#e6e9ef;--sub:#8b93a7;--line:#262b36;--line2:#1f2530;--accent:#2E74D6;--hbg:rgba(15,18,24,.88);--okt:#2FBF8F;--infot:#6ea2ec;--errt:#E05D5D;--mut:#5c6473}
|
||||
body[data-theme="cobalt"]{--bg:#080d1c;--card:#111a33;--card2:#0b1226;--text:#e9ecfb;--sub:#8a93b8;--line:#1d2747;--line2:#161f3a;--accent:#4068FF;--hbg:rgba(8,13,28,.9);--okt:#2fd0a6;--infot:#8aa0ff;--errt:#f06a7a;--mut:#5a6390}
|
||||
body[data-theme="cupru"]{--bg:#15110b;--card:#211a12;--card2:#15110b;--text:#efe6d6;--sub:#a89a85;--line:#36291c;--line2:#281e14;--accent:#D98A3D;--hbg:rgba(21,17,11,.9);--okt:#67b98c;--infot:#dfa45c;--errt:#e2685a;--mut:#6d5f4c}
|
||||
body[data-theme="hartie"]{--bg:#f3efe6;--card:#fffdf7;--card2:#f3efe6;--text:#1e1a13;--sub:#6a6052;--line:#e2dccc;--line2:#ece6d9;--accent:#1F5FBF;--hbg:rgba(255,253,247,.92);--okt:#1c7d5d;--infot:#1F5FBF;--errt:#bd463c;--mut:#9a8f7d}
|
||||
.page{width:100%;max-width:1280px;margin:0 auto;background:var(--bg,#0f1218);color:var(--text,#e6e9ef);overflow:hidden;}
|
||||
a{text-decoration:none;}
|
||||
input[type=range]{-webkit-appearance:none;appearance:none;background:transparent;}
|
||||
input[type=range]::-webkit-slider-runnable-track{height:6px;border-radius:99px;background:var(--line,#262b36);}
|
||||
input[type=range]::-moz-range-track{height:6px;border-radius:99px;background:var(--line,#262b36);}
|
||||
input[type=range]::-webkit-slider-thumb{-webkit-appearance:none;appearance:none;width:18px;height:18px;margin-top:-6px;border-radius:99px;background:var(--accent,#2E74D6);cursor:pointer;border:none;}
|
||||
input[type=range]::-moz-range-thumb{width:18px;height:18px;border-radius:99px;background:var(--accent,#2E74D6);cursor:pointer;border:none;}
|
||||
|
||||
@media (max-width:900px){
|
||||
.lp-nav{display:none!important;}
|
||||
.lp-header{padding:0 18px!important;}
|
||||
.lp-h1{font-size:32px!important;line-height:1.1!important;}
|
||||
.page [style*="grid-template-columns"]{grid-template-columns:1fr!important;}
|
||||
.page [style*="padding:80px 40px"]{padding:48px 20px!important;}
|
||||
.page [style*="padding:0 40px 80px"]{padding:0 20px 48px!important;}
|
||||
.page [style*="padding:56px 40px 80px"]{padding:36px 20px 48px!important;}
|
||||
.page [style*="padding:44px"]{padding:28px!important;}
|
||||
.page [style*="padding:56px 40px"]{padding:40px 22px!important;}
|
||||
.page [style*="height:68px"]{height:60px!important;}
|
||||
.page [style*="gap:56px"]{gap:32px!important;}
|
||||
.page [style*="gap:48px"]{gap:28px!important;}
|
||||
}
|
||||
@media (max-width:560px){
|
||||
.lp-h1{font-size:27px!important;}
|
||||
.page [style*="padding:10px 40px"]{padding:10px 18px!important;}
|
||||
.lp-header{padding:0 12px!important;}
|
||||
#theme-label{display:none!important;}
|
||||
.lp-hactions{gap:8px!important;}
|
||||
.lp-hactions button{height:38px!important;padding:0 11px!important;font-size:13px!important;}
|
||||
}
|
||||
@media (max-width:430px){
|
||||
.lp-hactions a.auth-login-link{display:none!important;}
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body data-theme="grafit">
|
||||
<script>try{var _t=localStorage.getItem('lp-theme');if(_t&&['grafit','cobalt','cupru','hartie'].indexOf(_t)>=0)document.body.setAttribute('data-theme',_t);}catch(e){}</script>
|
||||
<main class="page">
|
||||
<!-- HEADER -->
|
||||
<div class="lp-header" style="position:sticky;top:0;display:flex;align-items:center;justify-content:space-between;padding:0 40px;height:68px;background:var(--hbg,rgba(15,18,24,.88));backdrop-filter:blur(8px);border-bottom:1px solid var(--line,#262b36);z-index:5;">
|
||||
<div style="display:flex;align-items:center;gap:14px;">
|
||||
<img src="/static/romfast_logo.png" alt="ROMFAST" style="height:38px;width:auto;display:block;" />
|
||||
<div style="display:flex;flex-direction:column;line-height:1.05;">
|
||||
<span style="font:700 17px var(--font-ui);letter-spacing:-.01em;color:var(--text,#e6e9ef);">ROA AUTOPASS</span>
|
||||
<span style="font:500 11px var(--font-ui);letter-spacing:.04em;color:var(--sub,#8b93a7);">Gateway RAR</span>
|
||||
</div>
|
||||
</div>
|
||||
<div style="display:flex;align-items:center;gap:28px;">
|
||||
<div class="lp-nav" style="display:flex;gap:28px;font:500 14px var(--font-ui);color:var(--sub,#8b93a7);">
|
||||
<a href="#cum-functioneaza" style="color:inherit;text-decoration:none;">Cum funcționează</a><a href="#api" style="color:inherit;text-decoration:none;">API</a><a href="#pret" style="color:inherit;text-decoration:none;">Preț</a>
|
||||
</div>
|
||||
<div class="lp-hactions" style="display:flex;align-items:center;gap:12px;">
|
||||
<button data-act="theme" style="display:flex;align-items:center;gap:8px;height:40px;padding:0 13px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--sub,#8b93a7);font:500 13px var(--font-ui);cursor:pointer;">
|
||||
<svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg>
|
||||
<span id="theme-label">Grafit</span>
|
||||
</button>
|
||||
<a href="/login" class="auth-login-link" style="display:inline-flex;align-items:center;height:44px;padding:0 18px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--text,#e6e9ef);font:500 14px var(--font-ui);cursor:pointer;transition:transform .18s ease, background .18s ease, border-color .18s ease;text-decoration:none;" style-hover="background:#1F9D5C;border-color:#1F9D5C;color:#fff;transform:translateY(-1px)">Autentificare</a>
|
||||
<button data-act="auth" data-tab="register" style="height:44px;padding:0 18px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 14px var(--font-ui);cursor:pointer;transition:transform .2s cubic-bezier(.2,.8,.2,1), background .2s ease, box-shadow .2s ease;" style-hover="background:#16864a;border-color:#16864a;transform:translateY(-1px)">Creează cont</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- HERO -->
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:56px;align-items:center;padding:80px 40px 72px;">
|
||||
<div>
|
||||
<h1 class="lp-h1" style="font:700 50px/1.06 var(--font-ui);letter-spacing:-.025em;margin:0 0 20px;color:var(--text,#e6e9ef);">Declară prestațiile la RAR AUTOPASS, automat</h1>
|
||||
<p style="font:400 17px/1.6 var(--font-ui);color:var(--sub,#8b93a7);margin:0 0 24px;max-width:480px;">Încarci fișierul tău așa cum îl ai, potrivești o dată operațiile cu codurile RAR, și trimitem prestațiile la RAR AUTOPASS în locul tău. Fără tastat câmp cu câmp.</p>
|
||||
<div style="margin-bottom:32px;">
|
||||
<p style="display:flex;align-items:center;gap:8px;font:700 16px var(--font-ui);color:var(--text,#e6e9ef);margin:0;"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="#1F9D5C" stroke-width="2.6" style="flex-shrink:0;"><path d="M20 6L9 17l-5-5"/></svg><span><span style="color:#1F9D5C;">Gratuit</span> până la 60 de trimiteri/lună</span></p>
|
||||
</div>
|
||||
<div style="display:flex;gap:12px;margin-bottom:22px;">
|
||||
<button data-act="auth" data-tab="register" style="height:50px;padding:0 26px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 15px var(--font-ui);cursor:pointer;transition:transform .2s cubic-bezier(.2,.8,.2,1), background .2s ease, box-shadow .2s ease;" style-hover="background:#16864a;border-color:#16864a;transform:translateY(-1px)">Creează cont gratuit</button>
|
||||
<button style="height:50px;padding:0 24px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--text,#e6e9ef);font:500 15px var(--font-ui);cursor:pointer;transition:transform .18s ease, background .18s ease, border-color .18s ease;" style-hover="background:#1F9D5C;border-color:#1F9D5C;color:#fff;transform:translateY(-1px)">Vezi cum funcționează</button>
|
||||
</div>
|
||||
<div style="display:flex;align-items:center;gap:14px;font:400 13px var(--font-ui);color:var(--sub,#8b93a7);flex-wrap:wrap;">
|
||||
<span style="display:flex;align-items:center;gap:6px;"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2"><path d="M20 6L9 17l-5-5"/></svg>Conform Legii 142/2023</span>
|
||||
<span style="color:var(--line,#262b36);">·</span>
|
||||
<span style="display:flex;align-items:center;gap:6px;"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="1.7"><rect x="5" y="11" width="14" height="9" rx="1.5"/><path d="M8 11V8a4 4 0 0 1 8 0v3"/></svg>Datele tale criptate</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Dashboard mockup -->
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:12px;box-shadow:0 24px 60px -20px rgba(0,0,0,.6);overflow:hidden;">
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;padding:16px 18px;border-bottom:1px solid var(--line,#262b36);">
|
||||
<div>
|
||||
<div style="font:700 14px var(--font-ui);color:var(--text,#e6e9ef);">Trimiteri RAR AUTOPASS</div>
|
||||
<div style="font:400 12px var(--font-mono);color:var(--sub,#8b93a7);margin-top:2px;">Service Auto Vâlcea · 28 iun 2026</div>
|
||||
</div>
|
||||
<div style="display:flex;gap:8px;">
|
||||
<div style="display:flex;align-items:center;gap:5px;padding:4px 9px;border-radius:99px;background:color-mix(in srgb,#2FBF8F 13%,transparent);font:500 11px var(--font-ui);color:var(--okt,#2FBF8F);"><span style="width:6px;height:6px;border-radius:99px;background:#2FBF8F;"></span>Live</div>
|
||||
</div>
|
||||
</div>
|
||||
<div style="display:flex;gap:10px;padding:14px 18px;border-bottom:1px solid var(--line,#262b36);">
|
||||
<div style="flex:1;background:var(--card2,#0f1218);border:1px solid var(--line,#262b36);border-radius:8px;padding:10px 12px;"><div style="font:700 20px var(--font-ui);color:var(--text,#e6e9ef);">847</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">Trimise luna asta</div></div>
|
||||
<div style="flex:1;background:var(--card2,#0f1218);border:1px solid var(--line,#262b36);border-radius:8px;padding:10px 12px;"><div style="font:700 20px var(--font-ui);color:var(--accent,#2E74D6);">12</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">În coadă</div></div>
|
||||
<div style="flex:1;background:var(--card2,#0f1218);border:1px solid var(--line,#262b36);border-radius:8px;padding:10px 12px;"><div style="font:700 20px var(--font-ui);color:#E05D5D;">2</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">De corectat</div></div>
|
||||
</div>
|
||||
<div style="padding:6px 0;">
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;padding:11px 18px;border-bottom:1px solid var(--line2,#1f2530);">
|
||||
<div><div style="font:500 13px var(--font-mono);color:var(--text,#e6e9ef);">WBA8E9...K7F2</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">Inspecție tehnică · 09:42</div></div>
|
||||
<div style="display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;background:color-mix(in srgb,#2FBF8F 13%,transparent);font:500 12px var(--font-ui);color:var(--okt,#2FBF8F);"><span style="width:6px;height:6px;border-radius:99px;background:#2FBF8F;"></span>Trimis</div>
|
||||
</div>
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;padding:11px 18px;border-bottom:1px solid var(--line2,#1f2530);">
|
||||
<div><div style="font:500 13px var(--font-mono);color:var(--text,#e6e9ef);">WVWZZZ...3M1</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">Revizie periodică · 09:38</div></div>
|
||||
<div style="display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;background:color-mix(in srgb,var(--accent,#2E74D6) 14%,transparent);font:500 12px var(--font-ui);color:var(--infot,#6ea2ec);"><span style="width:6px;height:6px;border-radius:99px;background:var(--accent,#2E74D6);"></span>În coadă</div>
|
||||
</div>
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;padding:11px 18px;border-bottom:1px solid var(--line2,#1f2530);">
|
||||
<div><div style="font:500 13px var(--font-mono);color:var(--text,#e6e9ef);">VF1RFB...A88</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">Sistem frânare · 09:31</div></div>
|
||||
<div style="display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;background:color-mix(in srgb,#E05D5D 14%,transparent);font:500 12px var(--font-ui);color:var(--errt,#E05D5D);"><span style="width:6px;height:6px;border-radius:99px;background:#E05D5D;"></span>Eroare VIN</div>
|
||||
</div>
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;padding:11px 18px;">
|
||||
<div><div style="font:500 13px var(--font-mono);color:var(--text,#e6e9ef);">ZAR937...C04</div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);">Schimb ulei · 09:24</div></div>
|
||||
<div style="display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;background:color-mix(in srgb,#2FBF8F 13%,transparent);font:500 12px var(--font-ui);color:var(--okt,#2FBF8F);"><span style="width:6px;height:6px;border-radius:99px;background:#2FBF8F;"></span>Trimis</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- PROBLEM + CALCULATOR (combinat) -->
|
||||
<div style="padding:80px 40px 40px;background:color-mix(in srgb,#E05D5D 6%,var(--bg,#0f1218));">
|
||||
<div style="text-align:center;max-width:760px;margin:0 auto 40px;">
|
||||
<h2 style="font:700 38px/1.14 var(--font-ui);letter-spacing:-.02em;margin:0 0 14px;color:var(--text,#e6e9ef);">Pentru fiecare comandă stai 2–3 minute pe RAR AUTOPASS.<br><span style="color:var(--errt,#E05D5D);">Minutele acelea sunt bani.</span></h2>
|
||||
<p style="font:400 16px/1.65 var(--font-ui);color:var(--sub,#8b93a7);margin:0;">VIN, cod prestație, kilometraj, dată, tip operație — câmp cu câmp, comandă cu comandă. La 20 de mașini pe zi pierzi aproape o oră. În fiecare zi. Mută cursorul la volumul service-ului tău și vezi cât te costă.</p>
|
||||
</div>
|
||||
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:24px;align-items:stretch;">
|
||||
<!-- STANGA: formularul RAR AUTOPASS -->
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:14px;padding:24px;display:flex;flex-direction:column;">
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;margin-bottom:16px;"><span style="font:500 12px var(--font-mono);color:var(--sub,#8b93a7);">RAR AUTOPASS · prestație nouă</span><span style="display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;background:color-mix(in srgb,#E05D5D 14%,transparent);color:var(--errt,#E05D5D);font:600 12px var(--font-mono);"><svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="13" r="8"/><path d="M12 9v4l2.5 2.5M9 2h6"/></svg>02:34</span></div>
|
||||
<div style="display:flex;flex-direction:column;gap:10px;">
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:10px;">
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">VIN</div><div style="height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);font:400 12px var(--font-mono);color:var(--text,#e6e9ef);overflow:hidden;white-space:nowrap;">U1234567890123456</div></div>
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">Confirmă VIN</div><div style="height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);font:400 12px var(--font-mono);color:var(--text,#e6e9ef);overflow:hidden;white-space:nowrap;">U1234567890123456</div></div>
|
||||
</div>
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:10px;">
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">Data prestației</div><div style="height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);font:400 12px var(--font-mono);color:var(--text,#e6e9ef);">2026-06-22</div></div>
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">Număr înmatriculare</div><div style="height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);font:400 12px var(--font-mono);color:var(--text,#e6e9ef);">CT88NOE</div></div>
|
||||
</div>
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">Observații</div><div style="height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);font:400 12px var(--font-ui);color:var(--text,#e6e9ef);overflow:hidden;white-space:nowrap;">REVIZIE; SCHIMBARE PLĂCUȚE FRÂNĂ</div></div>
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">Prestații</div><div style="min-height:30px;display:flex;align-items:center;gap:6px;flex-wrap:wrap;padding:3px 6px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);"><span style="display:inline-flex;align-items:center;gap:4px;padding:3px 8px;border-radius:5px;background:color-mix(in srgb,var(--accent,#2E74D6) 18%,transparent);color:var(--accent,#2E74D6);font:500 10px var(--font-ui);"><span style="opacity:.7;">×</span>REVIZIE PERIODICĂ</span><span style="display:inline-flex;align-items:center;gap:4px;padding:3px 8px;border-radius:5px;background:color-mix(in srgb,var(--accent,#2E74D6) 18%,transparent);color:var(--accent,#2E74D6);font:500 10px var(--font-ui);"><span style="opacity:.7;">×</span>ÎNTREȚINERE</span></div></div>
|
||||
<div><div style="font:400 11px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:4px;">Valoarea citită a odometrului</div><div style="height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);font:400 12px var(--font-mono);color:var(--text,#e6e9ef);">39000</div></div>
|
||||
</div>
|
||||
<button style="margin-top:14px;height:34px;padding:0 14px;border-radius:6px;background:color-mix(in srgb,var(--accent,#2E74D6) 40%,var(--card2,#0f1218));border:none;color:#fff;opacity:.55;font:600 12px var(--font-ui);cursor:not-allowed;align-self:flex-start;">Salvează Prezentarea</button>
|
||||
<div style="margin-top:auto;padding-top:12px;font:400 12px var(--font-ui);color:var(--sub,#8b93a7);text-align:center;">se repetă pentru fiecare comandă · zi de zi</div>
|
||||
</div>
|
||||
|
||||
<!-- DREAPTA: calculatorul (slidere + cifre) -->
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:14px;padding:34px;display:flex;flex-direction:column;">
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:28px;margin-bottom:28px;">
|
||||
<div>
|
||||
<div style="display:flex;align-items:baseline;justify-content:space-between;margin-bottom:10px;"><span style="font:500 13px var(--font-ui);color:var(--text,#e6e9ef);">Trimiteri/lună</span><span style="font:700 24px var(--font-ui);letter-spacing:-.02em;color:var(--accent,#2E74D6);" id="out-pres">100</span></div>
|
||||
<input type="range" min="50" max="1500" step="10" value="100" id="calc-pres" data-act="calc" style="width:100%;height:6px;accent-color:var(--accent,#2E74D6);cursor:pointer;" />
|
||||
</div>
|
||||
<div>
|
||||
<div style="display:flex;align-items:baseline;justify-content:space-between;margin-bottom:10px;"><span style="font:500 13px var(--font-ui);color:var(--text,#e6e9ef);">Manoperă</span><span style="font:700 24px var(--font-ui);letter-spacing:-.02em;color:var(--accent,#2E74D6);"><span id="out-rate">60</span><span style="font:500 12px var(--font-ui);color:var(--sub,#8b93a7);"> lei/h</span></span></div>
|
||||
<input type="range" min="30" max="200" step="5" value="60" id="calc-rate" data-act="calc" style="width:100%;height:6px;accent-color:var(--accent,#2E74D6);cursor:pointer;" />
|
||||
</div>
|
||||
</div>
|
||||
<div style="background:color-mix(in srgb,#E05D5D 9%,var(--card2,#0f1218));border:1px solid color-mix(in srgb,#E05D5D 28%,var(--line,#262b36));border-radius:10px;padding:22px 24px;">
|
||||
<div style="font:600 11px var(--font-ui);color:var(--errt,#E05D5D);letter-spacing:.08em;text-transform:uppercase;margin-bottom:14px;">Pierdut pe raportare manuală</div>
|
||||
<div style="display:grid;grid-template-columns:repeat(4,1fr);gap:14px;">
|
||||
<div><div style="font:700 30px var(--font-ui);letter-spacing:-.02em;color:var(--errt,#E05D5D);"><span data-calc="leiMonth">0</span></div><div style="font:400 12px var(--font-ui);color:var(--sub,#8b93a7);margin-top:2px;">lei/lună</div></div>
|
||||
<div><div style="font:700 30px var(--font-ui);letter-spacing:-.02em;color:var(--errt,#E05D5D);"><span data-calc="hMonth">0</span></div><div style="font:400 12px var(--font-ui);color:var(--sub,#8b93a7);margin-top:2px;">ore/lună</div></div>
|
||||
<div><div style="font:700 30px var(--font-ui);letter-spacing:-.02em;color:var(--errt,#E05D5D);"><span data-calc="leiYear">0</span></div><div style="font:400 12px var(--font-ui);color:var(--sub,#8b93a7);margin-top:2px;">lei/an</div></div>
|
||||
<div><div style="font:700 30px var(--font-ui);letter-spacing:-.02em;color:var(--errt,#E05D5D);"><span data-calc="days">0</span></div><div style="font:400 12px var(--font-ui);color:var(--sub,#8b93a7);margin-top:2px;">zile/an</div></div>
|
||||
</div>
|
||||
</div>
|
||||
<div style="margin-top:20px;padding-top:18px;border-top:1px solid var(--line,#262b36);">
|
||||
<div style="display:flex;align-items:center;gap:9px;font:600 14px var(--font-ui);color:var(--okt,#2FBF8F);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.2" style="flex-shrink:0;"><path d="M20 6L9 17l-5-5"/></svg>Cu ROA AUTOPASS: câteva secunde pentru tot lotul</div>
|
||||
<div style="font:400 13px/1.55 var(--font-ui);color:var(--sub,#8b93a7);margin-top:6px;">Recuperezi ~<span data-calc="leiMonth">0</span> lei/lună și timpul îl pui pe mașini, nu pe formulare.</div>
|
||||
</div>
|
||||
<div style="margin-top:14px;display:flex;align-items:center;gap:8px;font:400 12px var(--font-ui);color:var(--mut,#5c6473);"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7"><circle cx="12" cy="13" r="8"/><path d="M12 9v4l2.5 2.5M9 2h6"/></svg>Estimat la ~2,5 minute de tastat manual pentru fiecare trimitere.</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- LEGE / AMENZI -->
|
||||
<div style="padding:56px 40px 80px;">
|
||||
<div style="display:flex;gap:20px;align-items:flex-start;background:color-mix(in srgb,#E0A93B 9%,var(--card,#181c24));border:1px solid color-mix(in srgb,#E0A93B 34%,var(--line,#262b36));border-radius:12px;padding:26px 28px;">
|
||||
<div style="width:44px;height:44px;flex-shrink:0;border-radius:8px;background:color-mix(in srgb,#E0A93B 16%,transparent);display:flex;align-items:center;justify-content:center;color:#E0A93B;"><svg width="22" height="22" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><path d="M12 3l8 4v5c0 4.4-3.1 8.3-8 9.5C7.1 20.3 4 16.4 4 12V7l8-4z"/><path d="M9.5 12l1.8 1.8L15 10"/></svg></div>
|
||||
<div>
|
||||
<div style="font:700 16px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:7px;">Evită riscul amenzilor — transmite automat la RAR Auto-Pass</div>
|
||||
<p style="font:400 14px/1.65 var(--font-ui);color:var(--sub,#8b93a7);margin:0;">Conform <strong style="color:var(--text,#e6e9ef);font-weight:600;">Legii nr. 142/2023</strong> și <strong style="color:var(--text,#e6e9ef);font-weight:600;">OMTI nr. 210/2024</strong>, service-urile auto autorizate RAR trebuie să transmită, la finalizarea fiecărei prestații, informațiile obligatorii (VIN, kilometraj și, după caz, date privind intervențiile asupra odometrului și reparațiile rezultate din avarii grave). Nerespectarea obligației se sancționează cu amendă între <span style="color:var(--errt,#E05D5D);font-weight:600;">2.000 și 5.000 lei</span>, iar transmiterea unor informații eronate cu amendă între <span style="color:#E0A93B;font-weight:600;">1.000 și 2.000 lei</span>.</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- SOLVE -->
|
||||
<div id="cum-functioneaza" style="padding:80px 40px 40px;background:color-mix(in srgb,var(--accent,#2E74D6) 8%,var(--bg,#0f1218));border-top:1px solid var(--line,#262b36);border-bottom:1px solid var(--line,#262b36);">
|
||||
<div style="max-width:780px;margin:0 auto;text-align:center;">
|
||||
<h2 style="font:700 36px var(--font-ui);letter-spacing:-.02em;margin:0 0 18px;color:var(--text,#e6e9ef);">Nu trebuie să fii bun cu calculatorul</h2>
|
||||
<p style="font:400 19px/1.75 var(--font-ui);color:var(--sub,#8b93a7);margin:0 auto;max-width:660px;"><span style="color:var(--accent,#2E74D6);font-weight:700;text-decoration:underline;text-underline-offset:3px;">Încarci</span> fișierul CSV/XLSX (sau trimiți direct prin API). ROA AUTOPASS îți propune asocierile — tu le <span style="color:var(--accent,#2E74D6);font-weight:700;text-decoration:underline;text-underline-offset:3px;">confirmi sau corectezi</span> o singură dată — apoi le <span style="color:var(--accent,#2E74D6);font-weight:700;text-decoration:underline;text-underline-offset:3px;">trimitem</span> la RAR, iar tu doar <span style="color:var(--accent,#2E74D6);font-weight:700;text-decoration:underline;text-underline-offset:3px;">urmărești</span> pe ecran.</p>
|
||||
</div>
|
||||
<div style="text-align:center;max-width:880px;margin:38px auto 0;font:400 20px/1.6 var(--font-ui);color:var(--sub,#8b93a7);">
|
||||
<span style="text-decoration:line-through;text-decoration-color:var(--errt,#E05D5D);text-decoration-thickness:2px;">2–3 minute de tastat pentru fiecare comandă</span><span style="color:var(--text,#e6e9ef);font-weight:700;"> → câteva secunde pentru tot lotul.</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- API INTEGRATION -->
|
||||
<div id="api" style="padding:56px 40px 80px;">
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:14px;display:grid;grid-template-columns:1fr 1fr;gap:40px;padding:44px;align-items:center;">
|
||||
<div>
|
||||
<div style="display:inline-flex;align-items:center;gap:8px;padding:5px 11px;border-radius:99px;border:1px solid var(--line,#262b36);color:var(--sub,#8b93a7);font:500 12px var(--font-ui);margin-bottom:18px;">Pentru service-uri cu soft propriu</div>
|
||||
<h2 style="font:700 30px/1.15 var(--font-ui);letter-spacing:-.02em;margin:0 0 14px;color:var(--text,#e6e9ef);">Ai deja un soft de service? Conectează-l automat</h2>
|
||||
<p style="font:400 15px/1.65 var(--font-ui);color:var(--sub,#8b93a7);margin:0 0 24px;">Softul tău se poate conecta și direct la API-ul RAR Auto-Pass. Cu ROMFAST primești în plus asistență la maparea automată a operațiilor tale (prin mai multe metode) și salvarea mapărilor pentru trimiterile viitoare — totul printr-un singur apel, cu cheie API per cont.</p>
|
||||
<button style="height:44px;padding:0 20px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--text,#e6e9ef);font:500 14px var(--font-ui);cursor:pointer;display:inline-flex;align-items:center;gap:8px;transition:transform .18s ease, background .18s ease, border-color .18s ease;" style-hover="background:#1F9D5C;border-color:#1F9D5C;color:#fff;transform:translateY(-1px)">Vezi documentația API <svg width="15" height="15" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.8"><path d="M5 12h14M13 6l6 6-6 6"/></svg></button>
|
||||
</div>
|
||||
<div style="background:#0d1015;border:1px solid #262b36;border-radius:10px;overflow:hidden;">
|
||||
<div style="display:flex;align-items:center;gap:7px;padding:11px 14px;border-bottom:1px solid #262b36;">
|
||||
<span style="width:11px;height:11px;border-radius:99px;background:#E05D5D;"></span><span style="width:11px;height:11px;border-radius:99px;background:#E0A93B;"></span><span style="width:11px;height:11px;border-radius:99px;background:#2FBF8F;"></span>
|
||||
<span style="font:400 12px var(--font-mono);color:#8b93a7;margin-left:8px;">request.sh</span>
|
||||
</div>
|
||||
<pre style="margin:0;padding:18px;font:400 13px/1.7 var(--font-mono);color:#e6e9ef;overflow-x:auto;"><span style="color:#2FBF8F;">POST</span> /v1/prezentari
|
||||
<span style="color:#8b93a7;">Authorization:</span> <span style="color:#E0A93B;">rfak_••••••••</span>
|
||||
<span style="color:#8b93a7;">Content-Type:</span> application/json
|
||||
|
||||
{
|
||||
<span style="color:#6ea2ec;">"vin"</span>: <span style="color:#2FBF8F;">"WBA8E9C5..."</span>,
|
||||
<span style="color:#6ea2ec;">"cod_prestatie"</span>: <span style="color:#2FBF8F;">"ITP-01"</span>,
|
||||
<span style="color:#6ea2ec;">"odometru"</span>: <span style="color:#E0A93B;">142500</span>
|
||||
}</pre>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- TRIAL BENEFIT -->
|
||||
<div style="padding:0 40px 80px;">
|
||||
<div style="display:flex;align-items:center;gap:22px;background:color-mix(in srgb,#2FBF8F 9%,var(--card,#181c24));border:1px solid color-mix(in srgb,#2FBF8F 32%,var(--line,#262b36));border-radius:14px;padding:30px 34px;flex-wrap:wrap;">
|
||||
<div style="width:48px;height:48px;flex-shrink:0;border-radius:10px;background:color-mix(in srgb,#2FBF8F 16%,transparent);display:flex;align-items:center;justify-content:center;color:var(--okt,#2FBF8F);"><svg width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7"><path d="M12 2v4M12 18v4M2 12h4M18 12h4"/><circle cx="12" cy="12" r="4.5"/></svg></div>
|
||||
<div style="flex:1;min-width:240px;">
|
||||
<div style="font:700 19px var(--font-ui);letter-spacing:-.01em;color:var(--text,#e6e9ef);margin-bottom:5px;"><span style="color:var(--okt,#2FBF8F);">30 de zile Pro gratuit</span> la fiecare cont nou</div>
|
||||
<p style="font:400 14px/1.6 var(--font-ui);color:var(--sub,#8b93a7);margin:0;">Începi direct cu tot ce are planul Pro — import API, categorisire automată și suport rapid. După 30 de zile treci automat pe Gratuit, fără plată și fără întreruperi.</p>
|
||||
</div>
|
||||
<button data-act="auth" data-tab="register" data-plan="pro" style="height:48px;padding:0 24px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 14px var(--font-ui);cursor:pointer;white-space:nowrap;transition:transform .2s cubic-bezier(.2,.8,.2,1), background .2s ease, box-shadow .2s ease;" style-hover="background:#16864a;border-color:#16864a;transform:translateY(-1px)">Începe gratuit</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- PRICING -->
|
||||
<div id="pret" style="padding:0 40px 80px;">
|
||||
<div style="text-align:center;margin-bottom:44px;">
|
||||
<h2 style="font:700 34px var(--font-ui);letter-spacing:-.02em;margin:0 0 10px;color:var(--text,#e6e9ef);">Pentru un service mic, nu costă nimic</h2>
|
||||
<p style="font:400 15px var(--font-ui);color:var(--sub,#8b93a7);margin:0;">Alege planul potrivit volumului tău. Poți schimba sau anula oricând.</p>
|
||||
</div>
|
||||
<div style="display:grid;grid-template-columns:repeat(4,1fr);gap:16px;margin:0 auto;align-items:stretch;">
|
||||
<!-- Gratuit -->
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:12px;padding:26px 24px;display:flex;flex-direction:column;">
|
||||
<div style="font:700 17px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:8px;">Gratuit</div>
|
||||
<div style="display:flex;align-items:baseline;gap:5px;margin-bottom:4px;"><span style="font:700 32px var(--font-ui);letter-spacing:-.02em;color:var(--text,#e6e9ef);">0 lei</span><span style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);">/lună</span></div>
|
||||
<div style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:20px;">Până la 60 de trimiteri/lună</div>
|
||||
<div style="display:flex;flex-direction:column;gap:10px;margin-bottom:22px;flex:1;">
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Sugestii automate de operații RAR</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Mapare coloane cu salvare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Validare și anti-duplicat</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Trimiteri nelimitate</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Import prin API</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Categorisire automată, cu confirmare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Integrare în softul tău</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Suport email, în 48h</div>
|
||||
</div>
|
||||
<button data-act="auth" data-tab="register" data-plan="free" style="width:100%;height:46px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--text,#e6e9ef);font:500 14px var(--font-ui);cursor:pointer;transition:transform .18s ease, background .18s ease, border-color .18s ease;" style-hover="background:#1F9D5C;border-color:#1F9D5C;color:#fff;transform:translateY(-1px)">Creează cont gratuit</button>
|
||||
</div>
|
||||
<!-- Standard -->
|
||||
<div style="background:var(--card,#181c24);border:1.5px solid var(--accent,#2E74D6);border-radius:12px;padding:26px 24px;position:relative;display:flex;flex-direction:column;">
|
||||
<div style="position:absolute;top:-12px;left:20px;padding:4px 11px;border-radius:99px;background:var(--accent,#2E74D6);color:#fff;font:700 10px var(--font-ui);letter-spacing:.04em;text-transform:uppercase;">Popular</div>
|
||||
<div style="font:700 17px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:8px;">Standard</div>
|
||||
<div style="display:flex;align-items:baseline;gap:5px;margin-bottom:4px;"><span style="font:700 32px var(--font-ui);letter-spacing:-.02em;color:var(--text,#e6e9ef);">49 lei</span><span style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);">/lună</span><span style="font:400 12px var(--font-ui);color:var(--mut,#5c6473);">* fără TVA</span></div>
|
||||
<div style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:20px;">Trimiteri nelimitate</div>
|
||||
<div style="display:flex;flex-direction:column;gap:10px;margin-bottom:22px;flex:1;">
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Sugestii automate de operații RAR</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Mapare coloane cu salvare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Validare și anti-duplicat</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Trimiteri nelimitate</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Import prin API</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Categorisire automată, cu confirmare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Integrare în softul tău</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Suport email, maxim 24h</div>
|
||||
</div>
|
||||
<button style="width:100%;height:46px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 14px var(--font-ui);cursor:pointer;transition:transform .2s cubic-bezier(.2,.8,.2,1), background .2s ease, box-shadow .2s ease;" style-hover="background:#16864a;border-color:#16864a;transform:translateY(-1px)" data-act="auth" data-tab="register" data-plan="standard">Creează cont gratuit</button>
|
||||
</div>
|
||||
<!-- Pro -->
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:12px;padding:26px 24px;display:flex;flex-direction:column;">
|
||||
<div style="font:700 17px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:8px;">Pro</div>
|
||||
<div style="display:flex;align-items:baseline;gap:5px;margin-bottom:4px;"><span style="font:700 32px var(--font-ui);letter-spacing:-.02em;color:var(--text,#e6e9ef);">59 lei</span><span style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);">/lună</span><span style="font:400 12px var(--font-ui);color:var(--mut,#5c6473);">* fără TVA</span></div>
|
||||
<div style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:20px;">Nelimitat + acces API</div>
|
||||
<div style="display:flex;flex-direction:column;gap:10px;margin-bottom:22px;flex:1;">
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Sugestii automate de operații RAR</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Mapare coloane cu salvare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Validare și anti-duplicat</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Trimiteri nelimitate</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Import prin API</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Categorisire automată, cu confirmare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--mut,#5c6473);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="var(--mut,#5c6473)" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M5 12h14"/></svg>Integrare în softul tău</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Suport email, maxim 8h</div>
|
||||
</div>
|
||||
<button style="width:100%;height:46px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--text,#e6e9ef);font:500 14px var(--font-ui);cursor:pointer;transition:transform .18s ease, background .18s ease, border-color .18s ease;" style-hover="background:#1F9D5C;border-color:#1F9D5C;color:#fff;transform:translateY(-1px)" data-act="auth" data-tab="register" data-plan="pro">Creează cont gratuit</button>
|
||||
</div>
|
||||
<!-- Premium -->
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:12px;padding:26px 24px;display:flex;flex-direction:column;">
|
||||
<div style="font:700 17px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:8px;">Premium</div>
|
||||
<div style="display:flex;align-items:baseline;gap:5px;margin-bottom:4px;"><span style="font:700 32px var(--font-ui);letter-spacing:-.02em;color:var(--text,#e6e9ef);">La cerere</span></div>
|
||||
<div style="font:400 13px var(--font-ui);color:var(--sub,#8b93a7);margin-bottom:20px;">Soluție personalizată</div>
|
||||
<div style="display:flex;flex-direction:column;gap:10px;margin-bottom:22px;flex:1;">
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Sugestii automate de operații RAR</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Mapare coloane cu salvare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Validare și anti-duplicat</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Trimiteri nelimitate</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Import prin API</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Categorisire automată, cu confirmare</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Integrare în softul tău</div>
|
||||
<div style="display:flex;gap:9px;font:400 13px/1.4 var(--font-ui);color:var(--text,#e6e9ef);"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;margin-top:1px;"><path d="M20 6L9 17l-5-5"/></svg>Suport telefonic + onboarding dedicat</div>
|
||||
</div>
|
||||
<button style="width:100%;height:46px;border-radius:6px;background:transparent;border:1px solid var(--line,#262b36);color:var(--text,#e6e9ef);font:500 14px var(--font-ui);cursor:pointer;transition:transform .18s ease, background .18s ease, border-color .18s ease;" style-hover="background:#1F9D5C;border-color:#1F9D5C;color:#fff;transform:translateY(-1px)" data-act="auth" data-tab="register" data-plan="premium">Creează cont gratuit</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- PRIVACY -->
|
||||
<div style="padding:80px 40px;border-top:1px solid var(--line,#262b36);">
|
||||
<div style="margin:0 auto;display:grid;grid-template-columns:minmax(240px,330px) 1fr;gap:48px;align-items:center;">
|
||||
<h2 style="font:700 30px/1.2 var(--font-ui);letter-spacing:-.02em;margin:0;color:var(--text,#e6e9ef);">Datele clienților tăi nu devin marfă</h2>
|
||||
<div style="display:flex;flex-wrap:wrap;">
|
||||
<div style="flex:1;min-width:200px;padding:4px 28px;border-left:1px solid var(--line,#262b36);">
|
||||
<div style="font:700 16px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:7px;">Reținem doar strict necesarul</div>
|
||||
<div style="font:400 14px/1.55 var(--font-ui);color:var(--sub,#8b93a7);">Doar datele de care e nevoie ca să trimitem la RAR — nimic adunat în plus, nici la conturile gratuite.</div>
|
||||
</div>
|
||||
<div style="flex:1;min-width:200px;padding:4px 28px;border-left:1px solid var(--line,#262b36);">
|
||||
<div style="font:700 16px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:7px;">Doar pentru scopul declarat</div>
|
||||
<div style="font:400 14px/1.55 var(--font-ui);color:var(--sub,#8b93a7);">Maparea și trimiterea la RAR. Nu le vindem și nu le dăm mai departe.</div>
|
||||
</div>
|
||||
<div style="flex:1;min-width:200px;padding:4px 28px;border-left:1px solid var(--line,#262b36);">
|
||||
<div style="font:700 16px var(--font-ui);color:var(--text,#e6e9ef);margin-bottom:7px;">Se șterg la 3 luni</div>
|
||||
<div style="font:400 14px/1.55 var(--font-ui);color:var(--sub,#8b93a7);">Automat, fără să ceri — sau chiar acum, cu un singur click.</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- AUTH / REGISTER -->
|
||||
<div id="inregistrare" style="padding:80px 40px;border-top:1px solid var(--line,#262b36);background:color-mix(in srgb,var(--accent,#2E74D6) 5%,var(--bg,#0f1218));">
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:56px;margin:0 auto;align-items:center;">
|
||||
<div>
|
||||
<div style="font:500 13px var(--font-ui);color:var(--accent,#2E74D6);letter-spacing:.1em;text-transform:uppercase;margin-bottom:14px;">Creează cont</div>
|
||||
<h2 style="font:700 34px/1.15 var(--font-ui);letter-spacing:-.02em;margin:0 0 14px;color:var(--text,#e6e9ef);">Creează cont în 2 minute și declară azi la RAR</h2>
|
||||
<p style="font:400 16px/1.6 var(--font-ui);color:var(--sub,#8b93a7);margin:0 0 24px;">Te înregistrezi gratuit. Imediat poți încărca primul fișier sau conecta softul de service.</p>
|
||||
<div style="display:flex;flex-direction:column;gap:12px;">
|
||||
<div style="display:flex;gap:10px;align-items:center;font:400 15px var(--font-ui);color:var(--text,#e6e9ef);"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;"><path d="M20 6L9 17l-5-5"/></svg>Pro gratuit 30 de zile, apoi automat pe Gratuit</div>
|
||||
<div style="display:flex;gap:10px;align-items:center;font:400 15px var(--font-ui);color:var(--text,#e6e9ef);"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;"><path d="M20 6L9 17l-5-5"/></svg>Conform Legii 142/2023 și OMTI 210/2024</div>
|
||||
<div style="display:flex;gap:10px;align-items:center;font:400 15px var(--font-ui);color:var(--text,#e6e9ef);"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="#2FBF8F" stroke-width="2" style="flex-shrink:0;"><path d="M20 6L9 17l-5-5"/></svg>Datele cu caracter personal criptate (GDPR)</div>
|
||||
</div>
|
||||
</div>
|
||||
<div style="background:var(--card,#181c24);border:1px solid var(--line,#262b36);border-radius:14px;padding:32px;box-shadow:0 20px 50px -24px rgba(0,0,0,.5);">
|
||||
<div style="display:flex;gap:28px;border-bottom:1px solid var(--line,#262b36);margin-bottom:24px;">
|
||||
<button type="button" data-act="tab" data-tab="register" class="auth-tab is-active" style="position:relative;background:none;border:none;padding:0 0 12px;font:700 15px var(--font-ui);color:var(--text,#e6e9ef);cursor:pointer;">Creează cont<span class="tab-underline" style="position:absolute;left:0;right:0;bottom:-1px;height:2px;background:var(--accent,#2E74D6);"></span></button>
|
||||
<button type="button" data-act="tab" data-tab="login" class="auth-tab" style="position:relative;background:none;border:none;padding:0 0 12px;font:700 15px var(--font-ui);color:var(--sub,#8b93a7);cursor:pointer;">Autentificare<span class="tab-underline" style="position:absolute;left:0;right:0;bottom:-1px;height:2px;background:var(--accent,#2E74D6);"></span></button>
|
||||
</div>
|
||||
<form method="post" action="/signup" data-pane="register">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<label style="display:block;margin-bottom:14px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">Companie</span><input type="text" name="name" required placeholder="SC Service Auto SRL" style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-ui);outline:none;" /></label>
|
||||
<label style="display:block;margin-bottom:14px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">CUI</span><input type="text" name="cui" required placeholder="RO12345678" style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-mono);outline:none;" /></label>
|
||||
<label style="display:block;margin-bottom:14px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">Email</span><input type="email" name="email" required placeholder="nume@service.ro" style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-ui);outline:none;" /></label>
|
||||
<label style="display:block;margin-bottom:14px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">Parolă</span><input type="password" name="parola" required minlength="10" placeholder="Minim 10 caractere" style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-ui);outline:none;" /></label>
|
||||
<label style="display:block;margin-bottom:16px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">Pachet ales</span><select id="plan-select" name="plan" style="width:100%;height:44px;padding:0 10px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-ui);outline:none;cursor:pointer;"><option value="free" selected>Gratuit — 0 lei/lună</option><option value="standard">Standard — 49 lei/lună</option><option value="pro">Pro — 59 lei/lună</option><option value="premium">Premium — la cerere</option></select></label>
|
||||
<label style="display:flex;gap:9px;align-items:flex-start;margin-bottom:18px;font:400 13px/1.5 var(--font-ui);color:var(--sub,#8b93a7);cursor:pointer;"><input type="checkbox" name="consent" value="1" required style="margin-top:2px;accent-color:var(--accent,#2E74D6);width:16px;height:16px;flex-shrink:0;" />Sunt de acord cu Termenii și cu prelucrarea datelor conform politicii de confidențialitate (GDPR).</label>
|
||||
<button type="submit" class="btn-green" style="width:100%;height:48px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 15px var(--font-ui);cursor:pointer;transition:transform .2s cubic-bezier(.2,.8,.2,1), background .2s ease, box-shadow .2s ease;">Creează cont gratuit</button>
|
||||
<div style="text-align:center;margin-top:14px;font:400 13px var(--font-ui);color:var(--sub,#8b93a7);">Ai deja cont? <a data-act="tab" data-tab="login" style="color:var(--accent,#2E74D6);font-weight:500;cursor:pointer;">Autentifică-te</a></div>
|
||||
</form>
|
||||
<form method="post" action="/login" data-pane="login" style="display:none;">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<label style="display:block;margin-bottom:14px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">Email</span><input type="email" name="email" required placeholder="nume@service.ro" style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-ui);outline:none;" /></label>
|
||||
<label style="display:block;margin-bottom:10px;"><span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--sub,#8b93a7);">Parolă</span><input type="password" name="parola" required placeholder="Parola ta" style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line,#262b36);border-radius:6px;background:var(--card2,#0f1218);color:var(--text,#e6e9ef);font:400 14px var(--font-ui);outline:none;" /></label>
|
||||
<div style="text-align:right;margin-bottom:18px;"><a href="/login" style="font:400 13px var(--font-ui);color:var(--accent,#2E74D6);cursor:pointer;">Ai uitat parola?</a></div>
|
||||
<button type="submit" class="btn-green" style="width:100%;height:48px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 15px var(--font-ui);cursor:pointer;transition:transform .2s cubic-bezier(.2,.8,.2,1), background .2s ease;">Autentificare</button>
|
||||
<div style="text-align:center;margin-top:14px;font:400 13px var(--font-ui);color:var(--sub,#8b93a7);">Nu ai cont? <a data-act="tab" data-tab="register" style="color:var(--accent,#2E74D6);font-weight:500;cursor:pointer;">Creează unul gratuit</a></div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- FOOTER -->
|
||||
<div style="border-top:1px solid var(--line,#262b36);padding:36px 40px;display:flex;align-items:center;justify-content:space-between;flex-wrap:wrap;gap:16px;">
|
||||
<div style="font:700 18px var(--font-ui);letter-spacing:-.02em;color:var(--text,#e6e9ef);">ROM<span style="color:var(--accent,#2E74D6);">FAST</span></div>
|
||||
<div style="display:flex;gap:26px;font:400 14px var(--font-ui);color:var(--sub,#8b93a7);">
|
||||
<span>Termeni</span><span>Confidențialitate / GDPR</span><span>Documentație API</span><span>Contact</span>
|
||||
</div>
|
||||
<div style="font:400 13px var(--font-ui);color:var(--mut,#5c6473);">© 2026 ROMFAST</div>
|
||||
</div>
|
||||
</main>
|
||||
<script>
|
||||
(function(){
|
||||
var THEMES=[['grafit','Grafit'],['cobalt','Cobalt'],['cupru','Cupru'],['hartie','Hârtie']];
|
||||
var body=document.body;
|
||||
function curIndex(){var t=body.getAttribute('data-theme');for(var i=0;i<THEMES.length;i++){if(THEMES[i][0]===t)return i;}return 0;}
|
||||
function applyTheme(i){i=((i%THEMES.length)+THEMES.length)%THEMES.length;body.setAttribute('data-theme',THEMES[i][0]);var l=document.getElementById('theme-label');if(l)l.textContent=THEMES[i][1];try{localStorage.setItem('lp-theme',THEMES[i][0]);}catch(e){}}
|
||||
applyTheme(curIndex());
|
||||
|
||||
// style-hover: framework-ul de design folosea atributul style-hover; il aplicam la hover.
|
||||
function parseStyle(str){var o={};str.split(';').forEach(function(p){var idx=p.indexOf(':');if(idx>0)o[p.slice(0,idx).trim()]=p.slice(idx+1).trim();});return o;}
|
||||
document.querySelectorAll('[style-hover]').forEach(function(el){
|
||||
var hov=parseStyle(el.getAttribute('style-hover'));var keys=Object.keys(hov);var saved={};
|
||||
el.addEventListener('mouseenter',function(){keys.forEach(function(k){saved[k]=el.style.getPropertyValue(k);el.style.setProperty(k,hov[k]);});});
|
||||
el.addEventListener('mouseleave',function(){keys.forEach(function(k){el.style.setProperty(k,saved[k]);});});
|
||||
});
|
||||
|
||||
// Calculator: cost raportare manuala (2,5 min/prestatie).
|
||||
var pres=document.getElementById('calc-pres'),rate=document.getElementById('calc-rate');
|
||||
var nf=new Intl.NumberFormat('ro-RO',{maximumFractionDigits:0});
|
||||
var nf1=new Intl.NumberFormat('ro-RO',{maximumFractionDigits:1});
|
||||
function recalc(){
|
||||
var p=+pres.value,r=+rate.value,minPer=2.5;
|
||||
var hMonth=(p*minPer)/60,leiMonth=hMonth*r;
|
||||
document.getElementById('out-pres').textContent=p;
|
||||
document.getElementById('out-rate').textContent=r;
|
||||
var map={leiMonth:nf.format(Math.round(leiMonth)),hMonth:nf.format(Math.round(hMonth)),leiYear:nf.format(Math.round(leiMonth*12)),days:nf.format(Math.round((hMonth*12)/8))};
|
||||
Object.keys(map).forEach(function(k){document.querySelectorAll('[data-calc="'+k+'"]').forEach(function(n){n.textContent=map[k];});});
|
||||
}
|
||||
if(pres&&rate){pres.addEventListener('input',recalc);rate.addEventListener('input',recalc);recalc();}
|
||||
|
||||
// Tab-uri autentificare/inregistrare.
|
||||
function setTab(tab){
|
||||
document.querySelectorAll('[data-pane]').forEach(function(f){f.style.display=(f.getAttribute('data-pane')===tab)?'':'none';});
|
||||
document.querySelectorAll('.auth-tab').forEach(function(b){
|
||||
var on=b.getAttribute('data-tab')===tab;b.classList.toggle('is-active',on);
|
||||
b.style.color=on?'var(--text,#e6e9ef)':'var(--sub,#8b93a7)';
|
||||
var u=b.querySelector('.tab-underline');if(u)u.style.display=on?'':'none';
|
||||
});
|
||||
}
|
||||
setTab('register');
|
||||
|
||||
document.addEventListener('click',function(e){
|
||||
var t=e.target.closest('[data-act]');if(!t)return;
|
||||
var act=t.getAttribute('data-act');
|
||||
if(act==='theme'){applyTheme(curIndex()+1);}
|
||||
else if(act==='tab'){e.preventDefault();setTab(t.getAttribute('data-tab'));}
|
||||
else if(act==='auth'){
|
||||
e.preventDefault();
|
||||
setTab(t.getAttribute('data-tab')||'register');
|
||||
var plan=t.getAttribute('data-plan'),sel=document.getElementById('plan-select');
|
||||
if(plan&&sel)sel.value=plan;
|
||||
var a=document.getElementById('inregistrare');if(a)a.scrollIntoView({behavior:'smooth',block:'start'});
|
||||
}
|
||||
});
|
||||
})();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
@@ -1,28 +1,59 @@
|
||||
{% extends "base.html" %}
|
||||
{% block title %}Autentificare — Gateway RAR AUTOPASS{% endblock %}
|
||||
{% block title %}Autentificare — ROA AUTOPASS{% endblock %}
|
||||
{% block content %}
|
||||
<div class="card auth-card" style="max-width:400px;margin:40px auto;">
|
||||
<h2 style="margin-top:0;">Autentificare</h2>
|
||||
{# US-010 (PRD 5.16): /login — layout 2 coloane branduit.
|
||||
Stanga: logo + tagline + puncte de incredere.
|
||||
Dreapta: formular de autentificare (neschimbat: CSRF, POST /login, link signup).
|
||||
Pe mobil (<640px): se stivuiesc, partea dreapta (formular) iese prima. #}
|
||||
<div class="login-2col" style="max-width:860px; margin:32px auto;">
|
||||
{# Antet minimal deja randat in base.html (fara RAR dot, fara burger, fara account_name) #}
|
||||
<div class="login-shell">
|
||||
{# === Formular autentificare === #}
|
||||
<div class="login-form-col">
|
||||
<h3 style="font-size:var(--fs-xl); margin:0 0 4px;">Autentificare</h3>
|
||||
<p style="font-size:var(--fs-sm); color:var(--muted); margin:0 0 22px;">
|
||||
Intra in contul service-ului tau.
|
||||
</p>
|
||||
|
||||
{% if error %}
|
||||
<div class="banner" style="margin-bottom:12px;padding:8px 12px;">{{ error }}</div>
|
||||
{% endif %}
|
||||
{% if error %}
|
||||
<div class="banner" style="margin-bottom:14px; padding:8px 12px;">{{ error }}</div>
|
||||
{% endif %}
|
||||
|
||||
<form method="post" action="/login">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<p>
|
||||
<label>Email</label><br>
|
||||
<input type="email" name="email" required style="width:100%;">
|
||||
</p>
|
||||
<p>
|
||||
<label>Parola</label><br>
|
||||
<input type="password" name="parola" required style="width:100%;">
|
||||
</p>
|
||||
<button type="submit" style="width:100%;margin-top:8px;">Intra in cont</button>
|
||||
</form>
|
||||
<form method="post" action="/login">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<div class="camp-slim">
|
||||
<label for="lf-email">Email</label>
|
||||
<input id="lf-email" type="email" name="email" required autocomplete="email">
|
||||
</div>
|
||||
<div class="camp-slim" style="margin-bottom:14px;">
|
||||
<label for="lf-parola">Parola</label>
|
||||
<input id="lf-parola" type="password" name="parola" required autocomplete="current-password">
|
||||
</div>
|
||||
<button type="submit" class="btn-primary-full">Intra in cont</button>
|
||||
</form>
|
||||
|
||||
<p style="text-align:center;font-size:13px;margin-top:16px;">
|
||||
Cont nou? <a href="/signup">Inregistrare</a>
|
||||
</p>
|
||||
<p class="login-foot">
|
||||
Cont nou? <a href="/signup" style="color:var(--accent);">Inregistreaza service-ul</a>
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<style>
|
||||
/* US-010 PRD 5.16: layout /login profesional 2 coloane. */
|
||||
.login-shell {
|
||||
display:grid; grid-template-columns:1fr;
|
||||
border:1px solid var(--line); border-radius:16px; overflow:hidden;
|
||||
background:var(--card); max-width:460px; margin:0 auto;
|
||||
}
|
||||
.login-form-col { padding:40px 38px; display:flex; flex-direction:column; justify-content:center; }
|
||||
.btn-primary-full { width:100%; min-height:46px; font-family:var(--font-ui); font-size:var(--fs-md);
|
||||
font-weight:600; background:var(--accent); color:#fff; border:none;
|
||||
border-radius:8px; cursor:pointer; margin-top:4px; }
|
||||
.btn-primary-full:hover { filter:brightness(1.08); }
|
||||
.login-foot { text-align:center; font-size:var(--fs-sm); color:var(--muted); margin-top:18px; }
|
||||
@media (max-width:640px) {
|
||||
.login-form-col { padding:28px 22px; }
|
||||
}
|
||||
</style>
|
||||
{% endblock %}
|
||||
|
||||
@@ -37,33 +37,53 @@
|
||||
});
|
||||
</script>
|
||||
{% else %}
|
||||
<h2 style="margin-top:0;">Inregistrare cont nou</h2>
|
||||
<h2 style="margin-top:0;">Creează cont nou</h2>
|
||||
|
||||
{% if error %}
|
||||
<div class="banner" style="margin-bottom:12px;padding:8px 12px;">{{ error }}</div>
|
||||
{% endif %}
|
||||
|
||||
{# Format aliniat la formularul de inregistrare din landing (#inregistrare): aceleasi campuri,
|
||||
etichete, placeholder-uri si stil. Valorile `plan` = coduri tier (free/standard/pro/premium),
|
||||
normalizate server-side. #}
|
||||
<form method="post" action="/signup">
|
||||
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
||||
<p>
|
||||
<label>Companie <span style="color:var(--err)">*</span></label><br>
|
||||
<input type="text" name="name" value="{{ name or '' }}" required style="width:100%;">
|
||||
</p>
|
||||
<p>
|
||||
<label>CUI <span style="color:var(--muted);font-size:12px;">(optional)</span></label><br>
|
||||
<input type="text" name="cui" value="{{ cui or '' }}" style="width:100%;">
|
||||
</p>
|
||||
<p>
|
||||
<label>Email <span style="color:var(--err)">*</span></label><br>
|
||||
<input type="email" name="email" value="{{ email or '' }}" required style="width:100%;">
|
||||
</p>
|
||||
<p>
|
||||
<label>Parola <span style="color:var(--err)">*</span>
|
||||
<span style="color:var(--muted);font-size:12px;">(minim 10 caractere)</span>
|
||||
</label><br>
|
||||
<input type="password" name="parola" required style="width:100%;">
|
||||
</p>
|
||||
<button type="submit" style="width:100%;margin-top:8px;">Creeaza cont</button>
|
||||
<label style="display:block;margin-bottom:14px;">
|
||||
<span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--muted);">Companie</span>
|
||||
<input type="text" name="name" value="{{ name or '' }}" required placeholder="SC Service Auto SRL"
|
||||
style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line);border-radius:6px;background:var(--card2);color:var(--ink);font:400 14px var(--font-ui);outline:none;">
|
||||
</label>
|
||||
<label style="display:block;margin-bottom:14px;">
|
||||
<span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--muted);">CUI</span>
|
||||
<input type="text" name="cui" value="{{ cui or '' }}" required placeholder="RO12345678"
|
||||
style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line);border-radius:6px;background:var(--card2);color:var(--ink);font:400 14px var(--font-mono);outline:none;">
|
||||
</label>
|
||||
<label style="display:block;margin-bottom:14px;">
|
||||
<span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--muted);">Email</span>
|
||||
<input type="email" name="email" value="{{ email or '' }}" required placeholder="nume@service.ro"
|
||||
style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line);border-radius:6px;background:var(--card2);color:var(--ink);font:400 14px var(--font-ui);outline:none;">
|
||||
</label>
|
||||
<label style="display:block;margin-bottom:14px;">
|
||||
<span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--muted);">Parolă</span>
|
||||
<input type="password" name="parola" required minlength="10" placeholder="Minim 10 caractere"
|
||||
style="width:100%;height:44px;padding:0 12px;border:1px solid var(--line);border-radius:6px;background:var(--card2);color:var(--ink);font:400 14px var(--font-ui);outline:none;">
|
||||
</label>
|
||||
<label style="display:block;margin-bottom:16px;">
|
||||
<span style="display:block;margin-bottom:6px;font:500 13px var(--font-ui);color:var(--muted);">Pachet ales</span>
|
||||
<select name="plan"
|
||||
style="width:100%;height:44px;padding:0 10px;border:1px solid var(--line);border-radius:6px;background:var(--card2);color:var(--ink);font:400 14px var(--font-ui);outline:none;cursor:pointer;">
|
||||
<option value="free"{% if not plan or plan == 'free' %} selected{% endif %}>Gratuit — 0 lei/lună</option>
|
||||
<option value="standard"{% if plan == 'standard' %} selected{% endif %}>Standard — 49 lei/lună</option>
|
||||
<option value="pro"{% if plan == 'pro' %} selected{% endif %}>Pro — 59 lei/lună</option>
|
||||
<option value="premium"{% if plan == 'premium' %} selected{% endif %}>Premium — la cerere</option>
|
||||
</select>
|
||||
</label>
|
||||
<label style="display:flex;gap:9px;align-items:flex-start;margin-bottom:18px;font:400 13px/1.5 var(--font-ui);color:var(--muted);cursor:pointer;">
|
||||
<input type="checkbox" name="consent" value="1" required style="margin-top:2px;accent-color:var(--accent);width:16px;height:16px;flex-shrink:0;">
|
||||
Sunt de acord cu Termenii și cu prelucrarea datelor conform politicii de confidențialitate (GDPR).
|
||||
</label>
|
||||
<button type="submit"
|
||||
style="width:100%;height:48px;border-radius:6px;background:#1F9D5C;border:1px solid #1F9D5C;color:#fff;font:700 15px var(--font-ui);cursor:pointer;">Creează cont gratuit</button>
|
||||
</form>
|
||||
<p style="text-align:center;font-size:13px;margin-top:16px;">
|
||||
Ai deja cont? <a href="/login">Autentificare</a>
|
||||
|
||||
@@ -1,26 +1,23 @@
|
||||
"""Worker RAR — proces propriu (NU task asyncio in uvicorn; plan.md sect. 4).
|
||||
"""Worker RAR — proces propriu (NU task asyncio in uvicorn).
|
||||
|
||||
Bucla: heartbeat -> recupereaza orfane -> claim atomic -> login -> postPrezentare -> update.
|
||||
Ruleaza ca proces separat sub `restart: always` (docker compose).
|
||||
|
||||
T2 implementat:
|
||||
- claim atomic anti-race (BEGIN IMMEDIATE), respecta next_attempt_at (backoff).
|
||||
- reconciliere anti-duplicat pe raspuns pierdut: pe eroare tranzitorie/timeout SAU pe
|
||||
rand 'sending' orfan (worker mort mid-POST), interogheaza finalizate si match pe
|
||||
vin+dataPrestatie+odometruFinal; daca exista -> 'sent' (NU re-trimite).
|
||||
- retry/backoff exponential pe erori tranzitorii; peste worker_max_retries -> 'error' (banner).
|
||||
- retry/backoff exponential pe erori tranzitorii; peste worker_max_retries -> 'error'.
|
||||
- lease/timeout pe randuri 'sending' orfane.
|
||||
- re-login la token expirat (401 mid-sesiune) — JWT 30h, retry NU plafonat la 30h.
|
||||
|
||||
Creds per-cerere (plan sect. 5): fiecare submission poarta creds RAR CRIPTATE
|
||||
(rar_creds_enc). Worker-ul face login per CONT cu acele creds, cache-uieste JWT
|
||||
(30h) in memorie si STERGE creds-urile contului dupa primul login reusit. Token-ul
|
||||
in memorie acopera restul trimiterilor; la restart token-ul se pierde si contul
|
||||
re-loghează la urmatorul submission care aduce creds proaspete (degradare acceptata).
|
||||
Creds per-cerere: fiecare submission poarta creds RAR CRIPTATE (rar_creds_enc).
|
||||
Worker-ul face login per CONT cu acele creds, cache-uieste JWT (30h) in memorie si
|
||||
STERGE creds-urile contului dupa primul login reusit. Token-ul in memorie acopera
|
||||
restul trimiterilor; la restart token-ul se pierde si contul re-logheaza la urmatorul
|
||||
submission care aduce creds proaspete (degradare acceptata).
|
||||
Dev: `worker_use_test_creds` foloseste creds <test> cand submission-ul nu are enc.
|
||||
|
||||
Ce NU e inca: criptare PII payload at-rest (P2), b64Image mare pe disc (P2).
|
||||
|
||||
Pornire: python -m app.worker
|
||||
"""
|
||||
|
||||
@@ -37,7 +34,7 @@ import httpx
|
||||
from .. import errors
|
||||
from ..config import Settings, get_settings, load_test_credentials
|
||||
from ..crypto import decrypt_creds
|
||||
from ..db import get_connection, init_db, write_heartbeat
|
||||
from ..db import get_connection, init_db, read_heartbeat, write_heartbeat
|
||||
from ..observ import log_event, set_source
|
||||
from ..mapping import DEFAULT_ACCOUNT_ID, upsert_nomenclator
|
||||
from ..payload import build_rar_payload
|
||||
@@ -61,8 +58,8 @@ def _iso(dt: datetime) -> str:
|
||||
|
||||
|
||||
def _wlog(conn, tip: str, mesaj: str, *, nivel: str = "INFO", account_id=None, cod=None, context=None) -> None:
|
||||
"""Migrare print -> jurnal structurat (US-005): emite evenimentul (sursa=worker, dublu
|
||||
canal DB+fisier) SI pastreaza linia in stdout (operatorul tailuieste .run/worker.log)."""
|
||||
"""Emite evenimentul (sursa=worker, dublu canal DB+fisier) SI pastreaza linia in
|
||||
stdout (operatorul tailuieste .run/worker.log)."""
|
||||
print(f"[worker] {mesaj}", flush=True)
|
||||
log_event(tip, nivel=nivel, account_id=account_id, cod=cod, mesaj=mesaj, context=context,
|
||||
conn=conn, sursa="worker")
|
||||
@@ -84,17 +81,17 @@ def _is_transient(exc: Exception) -> bool:
|
||||
|
||||
# --- Operatii pe submissions ---
|
||||
|
||||
# Stari blocate ne-sent care primesc retentie proprie (US-013). Mai scurta decat
|
||||
# cele 90z ale `sent`: un blocat n-are valoare de audit ca o trimitere reusita.
|
||||
# Stari blocate ne-sent care primesc retentie proprie. Mai scurta decat cele 90z
|
||||
# ale `sent`: un blocat n-are valoare de audit ca o trimitere reusita.
|
||||
_BLOCKED_STATES = ("error", "needs_data", "needs_mapping")
|
||||
|
||||
|
||||
def mark(conn, submission_id: int, status: str, *, rar_status_code=None, rar_error=None, id_prezentare=None) -> None:
|
||||
if status == "sent":
|
||||
# T16: purge_after = sent + 90 zile (GDPR/L.142 retentie maxima).
|
||||
# purge_after = sent + 90 zile (GDPR/L.142 retentie maxima).
|
||||
purge_expr = "datetime('now', '+90 days')"
|
||||
elif status in _BLOCKED_STATES:
|
||||
# US-013: randurile blocate primesc si ele purge_after (altfel raman permanent).
|
||||
# Randurile blocate primesc si ele purge_after (altfel raman permanent).
|
||||
days = int(get_settings().blocked_retention_days)
|
||||
purge_expr = f"datetime('now', '+{days} days')"
|
||||
else:
|
||||
@@ -114,15 +111,15 @@ def mark(conn, submission_id: int, status: str, *, rar_status_code=None, rar_err
|
||||
)
|
||||
|
||||
|
||||
# T16: purge interval in secunde (odata pe ora, nu prea agresiv)
|
||||
# Purge interval in secunde (odata pe ora, nu prea agresiv)
|
||||
_PURGE_INTERVAL_S = 3600
|
||||
|
||||
|
||||
def purge_expired(conn) -> dict[str, int]:
|
||||
"""Sterge randurile expirate (purge_after < now).
|
||||
|
||||
T16/OV-5 + US-013/US-008: submissions `sent` SI blocate (error/needs_data/needs_mapping)
|
||||
expirate; import_batches expirate (import_rows via CASCADE); app_events expirate (jurnal).
|
||||
Submissions `sent` SI blocate (error/needs_data/needs_mapping) expirate;
|
||||
import_batches expirate (import_rows via CASCADE); app_events expirate (jurnal).
|
||||
EXCLUDE explicit `queued`/`sending` (randuri active — nu se purjeaza niciodata, chiar
|
||||
daca ar avea un purge_after rezidual; reactivarea il curata oricum).
|
||||
Intoarce {submissions_purged, batches_purged, events_purged}.
|
||||
@@ -174,7 +171,7 @@ def claim_one(conn) -> dict | None:
|
||||
"FROM submissions s LEFT JOIN accounts a ON a.id = s.account_id "
|
||||
"WHERE s.status='queued' "
|
||||
"AND (s.next_attempt_at IS NULL OR s.next_attempt_at <= ?) "
|
||||
# Gate pe stare de cont (5.5): doar 'active' trimite. Derivam defensiv din `active`
|
||||
# Gate pe stare de cont: doar 'active' trimite. Derivam defensiv din `active`
|
||||
# cand `status` lipseste (DB veche pre-migrare), pastrand active=1 <=> 'active'.
|
||||
"AND COALESCE(a.status, CASE WHEN COALESCE(a.active,1)=1 THEN 'active' ELSE 'pending' END) = 'active' "
|
||||
"ORDER BY s.id LIMIT 1",
|
||||
@@ -253,7 +250,7 @@ def process_one(conn, settings: Settings, rar: RarClient, token: str, claimed: d
|
||||
# RAR a raspuns DEFINITIV cu o eroare de procesare (ex. ORA-12899). NU e o
|
||||
# pierdere de raspuns ambigua -> NU reconcilia (recordul, daca exista la RAR,
|
||||
# e PARTIAL/rupt si nu trebuie marcat fals 'sent') si NU reincerca (acelasi
|
||||
# input va esua iar). Marcam 'error' cu mesajul real RAR. (Confirmat live 2026-06-23.)
|
||||
# input va esua iar). Marcam 'error' cu mesajul real RAR.
|
||||
detail = json.dumps(errors.eroare("RAR_EROARE_SERVER", cauza=exc.rar_message), ensure_ascii=False)
|
||||
mark(conn, sid, "error", rar_status_code=500, rar_error=detail)
|
||||
_wlog(conn, "submission_error", f"submission {sid} -> error (RAR 500): {exc.rar_message}",
|
||||
@@ -363,7 +360,7 @@ class AccountSessions:
|
||||
token = rar.login(creds["email"], creds["password"])
|
||||
except RarAuthError as exc:
|
||||
rar.close()
|
||||
# US-005: login esuat (401) — FARA email/parola (doar codul HTTP + contul).
|
||||
# Login esuat (401) — FARA email/parola (doar codul HTTP + contul).
|
||||
log_event("rar_login", nivel="WARNING", account_id=account_id,
|
||||
cod="RAR_CREDS_INVALIDE",
|
||||
mesaj=f"login RAR esuat (cont {account_id}): {exc.status_code or 401}",
|
||||
@@ -375,11 +372,11 @@ class AccountSessions:
|
||||
raise
|
||||
self._sessions[account_id] = (rar, token)
|
||||
write_heartbeat(conn, rar_login_ok=True, detail=f"login RAR ok (cont {account_id})")
|
||||
# US-005: login reusit (fara email/parola in clar — context curat).
|
||||
# Login reusit (fara email/parola in clar — context curat).
|
||||
log_event("rar_login", account_id=account_id, mesaj=f"login RAR ok (cont {account_id})",
|
||||
context={"rezultat": "ok", "http": 200}, conn=conn, sursa="worker")
|
||||
# Creds efemere pe submissions nu mai sunt necesare: JWT acopera retry-urile -> sterge.
|
||||
# GATE PURJARE (T1/Voce#5): sterge DOAR submissions.rar_creds_enc, NU accounts.rar_creds_enc.
|
||||
# GATE PURJARE: sterge DOAR submissions.rar_creds_enc, NU accounts.rar_creds_enc.
|
||||
# Canal web: fallback exista in accounts -> purjarea e inofensiva (re-login dupa restart).
|
||||
# Canal API pur: purjarea e identica cu Treapta 1 (neatinsa).
|
||||
conn.execute(
|
||||
@@ -418,7 +415,7 @@ def _creds_for(claimed: dict, settings: Settings) -> dict | None:
|
||||
|
||||
|
||||
def _creds_from_account(conn, account_id: int) -> dict | None:
|
||||
"""Fallback T1/D4: crede RAR durabile per-cont din accounts.rar_creds_enc.
|
||||
"""Fallback: crede RAR durabile per-cont din accounts.rar_creds_enc.
|
||||
|
||||
Canal web nu are re-pusher. Cand submission n-are creds (sterse dupa primul login
|
||||
sau upload web fara creds), worker-ul re-citeste din cont si poate re-login oricand.
|
||||
@@ -431,24 +428,87 @@ def _creds_from_account(conn, account_id: int) -> dict | None:
|
||||
return None
|
||||
|
||||
|
||||
def _keepalive_target(conn, settings: Settings) -> tuple[int | None, dict | None]:
|
||||
"""Un cont cu creds durabile pentru login-ul de proba (sau creds <test> in dev).
|
||||
|
||||
Sare conturile ale caror creds NU se decripteaza sub cheia curenta — in dev
|
||||
`start.sh both` genereaza o cheie efemera noua la fiecare pornire, deci creds-urile
|
||||
durabile criptate sub cheia veche dau decrypt -> None. Fallback la creds <test>.
|
||||
"""
|
||||
rows = conn.execute(
|
||||
"SELECT id, rar_creds_enc FROM accounts "
|
||||
"WHERE rar_creds_enc IS NOT NULL ORDER BY id"
|
||||
).fetchall()
|
||||
for row in rows:
|
||||
creds = decrypt_creds(row["rar_creds_enc"])
|
||||
if creds and creds.get("email") and creds.get("password"):
|
||||
return row["id"], creds
|
||||
if settings.worker_use_test_creds:
|
||||
return DEFAULT_ACCOUNT_ID, load_test_credentials()
|
||||
return None, None
|
||||
|
||||
|
||||
def _maybe_keepalive(conn, settings: Settings, sessions: "AccountSessions", state: dict) -> None:
|
||||
"""Login de proba periodic cand coada e goala — verifica reachability RAR si
|
||||
pastreaza last_rar_login_ok proaspat ca dashboard-ul sa nu afiseze fals
|
||||
'RAR inaccesibil' doar din lipsa de trafic.
|
||||
|
||||
Sondeaza la cel mult o data pe interval (si pe succes, si pe esec): pe succes
|
||||
heartbeat-ul se reimprospateaza singur; pe esec real (RAR jos) last_rar_login_ok
|
||||
ramane vechi -> dashboard-ul degradeaza corect. Forteaza login real (invalideaza
|
||||
sesiunea cache-uita) ca proba sa fie autentica, nu un token vechi din cache.
|
||||
"""
|
||||
interval = settings.worker_rar_keepalive_interval_s
|
||||
if interval <= 0:
|
||||
return
|
||||
hb = read_heartbeat(conn)
|
||||
last = hb["last_rar_login_ok"] if hb else None
|
||||
if last:
|
||||
try:
|
||||
age = (datetime.now(timezone.utc) - datetime.fromisoformat(last)).total_seconds()
|
||||
if age < interval:
|
||||
return # login inca proaspat — nimic de facut
|
||||
except (ValueError, TypeError):
|
||||
pass
|
||||
now_ts = time.time()
|
||||
if now_ts - state["last_attempt"] < interval:
|
||||
return # deja am incercat recent (nu hartui RAR daca e jos)
|
||||
state["last_attempt"] = now_ts
|
||||
|
||||
account_id, creds = _keepalive_target(conn, settings)
|
||||
if account_id is None or not creds:
|
||||
return # niciun cont cu creds durabile — nimic de sondat
|
||||
sessions.invalidate(account_id) # forteaza login real, nu token din cache
|
||||
try:
|
||||
sessions.get_token(conn, account_id, creds) # reimprospateaza last_rar_login_ok la succes
|
||||
except RarAuthError:
|
||||
pass # creds invalide — deja logat in get_token (WARNING)
|
||||
except Exception as exc:
|
||||
# RAR indisponibil: last_rar_login_ok ramane vechi (corect). Nu propaga.
|
||||
log_event("rar_keepalive", nivel="WARNING", account_id=account_id,
|
||||
mesaj=f"keepalive RAR esuat (cont {account_id}): {type(exc).__name__}",
|
||||
context={"rezultat": "esuat"}, conn=conn, sursa="worker")
|
||||
|
||||
|
||||
def run() -> int:
|
||||
signal.signal(signal.SIGTERM, _stop)
|
||||
signal.signal(signal.SIGINT, _stop)
|
||||
|
||||
settings = get_settings()
|
||||
set_source("worker") # US-005: evenimentele worker-ului au sursa=worker (fisier app-worker.log)
|
||||
set_source("worker") # evenimentele worker-ului au sursa=worker (fisier app-worker.log)
|
||||
init_db()
|
||||
conn = get_connection()
|
||||
print(f"[worker] pornit (send_enabled={settings.worker_send_enabled}, env={settings.rar_env})", flush=True)
|
||||
|
||||
sessions = AccountSessions(settings)
|
||||
_last_purge_time: float = 0.0
|
||||
_keepalive_state = {"last_attempt": 0.0}
|
||||
|
||||
while _running:
|
||||
try:
|
||||
write_heartbeat(conn, detail=f"poll (queue={_queue_depth(conn)})")
|
||||
|
||||
# T16: purjare periodica (odata pe ora) — NU mai frecvent.
|
||||
# Purjare periodica (odata pe ora) — NU mai frecvent.
|
||||
now_ts = time.time()
|
||||
if now_ts - _last_purge_time >= _PURGE_INTERVAL_S:
|
||||
stats = purge_expired(conn)
|
||||
@@ -469,25 +529,28 @@ def run() -> int:
|
||||
# Nimic de trimis: recupereaza orfanii conturilor deja logate.
|
||||
for acct, rar, tok in sessions.active():
|
||||
recover_orphans(conn, settings, rar, tok, account_id=acct)
|
||||
# Login de proba periodic ca dashboard-ul sa nu afiseze fals
|
||||
# "RAR inaccesibil" din lipsa de trafic (vezi _maybe_keepalive).
|
||||
_maybe_keepalive(conn, settings, sessions, _keepalive_state)
|
||||
time.sleep(settings.worker_poll_interval_s)
|
||||
continue
|
||||
|
||||
sid = claimed["id"]
|
||||
account_id = claimed["account_id"]
|
||||
# T1/US-012: randul poarta creds proaspete (rar_creds_enc != NULL) — fie prima
|
||||
# trimitere a contului, fie o REACTIVARE dupa creds gresite. Invalidam sesiunea
|
||||
# RAR cache-uita ca un JWT vechi (30h) din parola GRESITA sa nu trimita cu ea,
|
||||
# Randul poarta creds proaspete (rar_creds_enc != NULL) — fie prima trimitere
|
||||
# a contului, fie o REACTIVARE dupa creds gresite. Invalidam sesiunea RAR
|
||||
# cache-uita ca un JWT vechi (30h) din parola GRESITA sa nu trimita cu ea,
|
||||
# ignorand corectia. Re-login imediat cu creds-urile noi.
|
||||
if claimed.get("creds_enc"):
|
||||
sessions.invalidate(account_id)
|
||||
# T1/D4: incearca creds din submission (canal API efemer), cu fallback la
|
||||
# Incearca creds din submission (canal API efemer), cu fallback la
|
||||
# accounts.rar_creds_enc (canal web durabil). Canal web n-are re-pusher.
|
||||
creds = _creds_for(claimed, settings) or _creds_from_account(conn, account_id)
|
||||
|
||||
try:
|
||||
token = sessions.get_token(conn, account_id, creds)
|
||||
except RarAuthError as exc:
|
||||
# Creds gresite (login 401): NU se face retry (plan, failure registry).
|
||||
# Creds gresite (login 401): NU se face retry.
|
||||
mark(conn, sid, "error", rar_status_code=401,
|
||||
rar_error=json.dumps(errors.eroare("RAR_CREDS_INVALIDE", cauza="credentiale RAR invalide"), ensure_ascii=False))
|
||||
# rar_login esuat e deja logat in get_token; aici doar tranzitia submission-ului.
|
||||
|
||||
@@ -9,15 +9,19 @@ services:
|
||||
api:
|
||||
build: .
|
||||
command: uvicorn app.main:app --host 0.0.0.0 --port 8010
|
||||
ports:
|
||||
- "8010:8010"
|
||||
volumes:
|
||||
- autopass-data:/data
|
||||
environment:
|
||||
AUTOPASS_DB_PATH: /data/autopass.db
|
||||
AUTOPASS_RAR_ENV: test
|
||||
# Override din environment (Dokploy) pentru staging; default = prod.
|
||||
AUTOPASS_RAR_ENV: ${AUTOPASS_RAR_ENV:-prod}
|
||||
# Fus orar RO pentru bucketarea contoarelor azi/luna (SQLite 'localtime', E7).
|
||||
TZ: ${TZ:-Europe/Bucharest}
|
||||
AUTOPASS_CREDS_KEY: ${AUTOPASS_CREDS_KEY:?seteaza AUTOPASS_CREDS_KEY in .env (vezi .env.example)}
|
||||
AUTOPASS_REQUIRE_API_KEY: ${AUTOPASS_REQUIRE_API_KEY:-false}
|
||||
# Embeddings (sugestie mapare, Stratul 2): prima cerere /mapari lazy-load-eaza
|
||||
# modelul ~230MB. Doar API-ul il incarca (worker-ul nu). Default off.
|
||||
AUTOPASS_EMBEDDINGS_ENABLED: ${AUTOPASS_EMBEDDINGS_ENABLED:-false}
|
||||
restart: always
|
||||
healthcheck:
|
||||
test: ["CMD", "python", "-c", "import urllib.request,sys; sys.exit(0 if urllib.request.urlopen('http://localhost:8010/healthz').status==200 else 1)"]
|
||||
@@ -32,10 +36,11 @@ services:
|
||||
- autopass-data:/data
|
||||
environment:
|
||||
AUTOPASS_DB_PATH: /data/autopass.db
|
||||
AUTOPASS_RAR_ENV: test
|
||||
AUTOPASS_RAR_ENV: ${AUTOPASS_RAR_ENV:-test}
|
||||
AUTOPASS_CREDS_KEY: ${AUTOPASS_CREDS_KEY:?seteaza AUTOPASS_CREDS_KEY in .env (vezi .env.example)}
|
||||
# Send dezactivat by default; activeaza pentru proba end-to-end.
|
||||
AUTOPASS_WORKER_SEND_ENABLED: "false"
|
||||
# Send activ by default (prod); pe staging seteaza AUTOPASS_WORKER_SEND_ENABLED=false
|
||||
# in Dokploy ca worker-ul sa NU trimita declaratii reale la RAR (Legea 142/2023).
|
||||
AUTOPASS_WORKER_SEND_ENABLED: ${AUTOPASS_WORKER_SEND_ENABLED:-true}
|
||||
restart: always
|
||||
depends_on:
|
||||
- api
|
||||
|
||||
File diff suppressed because one or more lines are too long
@@ -494,5 +494,18 @@ Record de test creat: `data.id = 68514` (FINALIZATA, permanent pe test). Confirm
|
||||
- header `User-Agent` obligatoriu (altfel 403 WAF).
|
||||
|
||||
Rămas neprobat: ce alte valori `sistemReparat` (în afară de `"null"`) acceptă (Open Q #2).
|
||||
|
||||
## Note integrare — planuri de cont (PRD 5.17)
|
||||
|
||||
**Poți dezvolta și testa pe planul Gratuit** fără niciun upgrade — `POST /v1/prezentari/valideaza`
|
||||
(dry-run) e permis pe orice plan, nu face enqueue și nu consumă cotă lunară. Primești același
|
||||
răspuns de validare (câmpuri, cod_prestatie, rezolvare operație) ca la trimiterea reală.
|
||||
|
||||
**Trimiterea reală cere planul Pro** (sau trial Pro activ): rutele `POST /v1/prezentari`,
|
||||
`POST /v1/import` și `POST /v1/import/{id}/commit` sunt gate-uite pe `api_access=True`
|
||||
(Pro/Premium). Un cont Free/Standard primește `403 PLAN_FARA_API`. Contactează-ne pentru upgrade.
|
||||
|
||||
Planul Gratuit are limită de **60 prezentări/lună** (indiferent de canal). La depășire: `422 PLAN_LIMITA_LUNARA`.
|
||||
Planul Pro nu are limită de volum. `GET /v1/nomenclator` rămâne public pe orice plan (exploatare pre-upgrade).
|
||||
</content>
|
||||
</invoke>
|
||||
|
||||
233
docs/design.md
Normal file
233
docs/design.md
Normal file
@@ -0,0 +1,233 @@
|
||||
# design.md — Sistemul de design Gateway RAR AUTOPASS
|
||||
|
||||
Sursa de adevar pentru deciziile vizuale ale aplicatiei web. **Orice plan de design
|
||||
(`/plan-design-review`, `/design-consultation`, `/design-review`) si orice modificare
|
||||
de UI trebuie sa porneasca de aici.** Unde un mockup sau o propunere difera de acest
|
||||
document, documentul are dreptate (sau se actualizeaza explicit, intr-un commit separat).
|
||||
|
||||
Limba UI: romana, fara diacritice in cod/atribute tehnice, cu diacritice acceptate in
|
||||
textul vizibil (fontul are `latin-ext`). Fara emoji.
|
||||
|
||||
CSS-ul traieste inline in `app/web/templates/base.html` (un singur `<style>`). Nu exista
|
||||
build step. Tokenii de mai jos sunt variabile CSS reale definite acolo.
|
||||
|
||||
---
|
||||
|
||||
## 1. Principii
|
||||
|
||||
1. **Compact, nu inghesuit.** Densitate mare de informatie utila, dar cu ritm si spatiu.
|
||||
Pe ecrane mici aratam ESENTIALUL, nu tot ce incape pe desktop. Restul intra in detaliu
|
||||
(modal) sau in linii secundare mici.
|
||||
2. **Compactarea e si pentru desktop.** Cand o componenta e mai lizibila compacta (ex.
|
||||
wizard-ul de import), forma compacta se aplica pe toate latimile, nu doar pe mobil.
|
||||
3. **Mobile-first ca verificare, nu ca scuza.** Orice ecran trebuie sa fie complet
|
||||
utilizabil la 360px latime, fara scroll orizontal de pagina si fara text rupt vertical.
|
||||
4. **Starea prin text + culoare, niciodata doar culoare** (accesibilitate; pill-uri cu
|
||||
eticheta umana, glife ✓/✗ cu text).
|
||||
5. **O singura zona de actiune dominanta pe ecran.** Un singur buton primar vizibil per
|
||||
context (ex. „Trimite la RAR"). Restul sunt secundare/ghost.
|
||||
6. **Tinte de atins generoase pe touch, sobre pe desktop.** Vezi scara de control.
|
||||
|
||||
---
|
||||
|
||||
## 2. Tokeni
|
||||
|
||||
### 2.1 Culoare (variabile CSS, 4 teme)
|
||||
|
||||
Paleta e definita pe `:root` (dark, default) si suprascrisa pe `[data-theme="light|petrol"]`.
|
||||
`auto` se rezolva la light/dark dupa `prefers-color-scheme`. **Nu folosi culori hardcodate;
|
||||
foloseste mereu variabilele.** Pentru tente, `color-mix(in srgb, var(--x) N%, transparent|var(--card))`.
|
||||
|
||||
| Token | Rol | dark | light | petrol |
|
||||
|-------|-----|------|-------|--------|
|
||||
| `--bg` | fundal pagina | `#0f1218` | `#f5f7fa` | `#0e1416` |
|
||||
| `--card` | suprafata card/meniu | `#181c24` | `#ffffff` | `#161e20` |
|
||||
| `--ink` | text principal | `#e6e9ef` | `#1a1d24` | `#e6e9ef` |
|
||||
| `--muted` | text secundar | `#8b93a7` | `#5c6473` | `#8b93a7` |
|
||||
| `--line` | borduri/separatoare | `#262b36` | `#e2e5ea` | `#232c2e` |
|
||||
| `--accent` | actiune primara / link | `#2E74D6` | `#1F66C9` | `#0E7C7B` |
|
||||
| `--ok` | succes / trimis | `#2FBF8F` | `#15803d` | `#2FBF8F` |
|
||||
| `--warn` | atentie / de verificat | `#E0A93B` | `#b45309` | `#E0A93B` |
|
||||
| `--err` | eroare / distructiv | `#E05D5D` | `#dc2626` | `#E05D5D` |
|
||||
|
||||
Accentul light (`#1F66C9`) e ales pentru contrast AA pe alb (5.51:1). Orice text colorat
|
||||
pe `--card` trebuie sa ramana >= 4.5:1 in toate cele 3 palete.
|
||||
|
||||
### 2.2 Tipografie
|
||||
|
||||
Font: **IBM Plex Sans** (UI), **IBM Plex Mono** (VIN, coduri, ID-uri). Self-hosted, `latin-ext`
|
||||
pentru diacritice, `font-display:swap`. Greutati disponibile: 400, 500, 700.
|
||||
|
||||
Scara (px / weight) — folosita consecvent, nu inventa marimi noi:
|
||||
|
||||
| Rol | size | weight | note |
|
||||
|-----|------|--------|------|
|
||||
| Titlu pagina (header) | 20 (desktop) / 17 (mobil) | 700 | letter-spacing -.01em |
|
||||
| Titlu sectiune / card | 15 | 600 | `h2.sec` |
|
||||
| Subtitlu / `h3` | 14 | 600 | |
|
||||
| Corp / controale | 14 | 400/500 | inputuri, butoane |
|
||||
| Eticheta camp, link card | 13 | 400/500 | `.cardlink`, label form |
|
||||
| Secundar / meta | 12 | 400 | text muted, sub-linii |
|
||||
| Micro (coduri, badge) | 11 | 500/700 | mono pentru coduri |
|
||||
|
||||
Numerele tabulare: `font-variant-numeric: tabular-nums` pe tabele (aliniere coloane).
|
||||
Coduri/VIN/ID: `font-family: "IBM Plex Mono"`.
|
||||
|
||||
### 2.3 Spatiere
|
||||
|
||||
Scara 4px: **4, 6, 8, 10, 12, 14, 16, 20, 24**. Padding card desktop `16px 20px`, mobil `16px`.
|
||||
Gap intre carduri `14–16px`. Gap intre controale pe o linie `8–12px`.
|
||||
|
||||
### 2.4 Radius
|
||||
|
||||
| Valoare | Uz |
|
||||
|---------|-----|
|
||||
| `6px` | controale: butoane, input, select |
|
||||
| `7–8px` | carduri-rand, meniuri, butoane icon |
|
||||
| `10px` | carduri de sectiune |
|
||||
| `12px` | modal (desktop) |
|
||||
| `99px` | pill-uri, badge-uri, bara de progres |
|
||||
|
||||
### 2.5 Elevatie
|
||||
|
||||
Plat implicit (border `1px solid var(--line)`). Umbra DOAR pentru elemente plutitoare:
|
||||
meniuri/kebab `0 8px 24px rgba(0,0,0,.18)`, modal `0 16px 48px rgba(0,0,0,.35)`.
|
||||
|
||||
---
|
||||
|
||||
## 3. Breakpoints
|
||||
|
||||
Un singur prag conceptual mobil la **768px**; un prag de densitate la **1024px**.
|
||||
|
||||
| Interval | Numit | Regula |
|
||||
|----------|-------|--------|
|
||||
| `>= 1024px` | desktop | layout complet; aplica si compactarile globale (wizard) |
|
||||
| `768–1024px` | tableta | **card-uri** pentru tabelele actionabile (Trimiteri, Preview, Mapari), UN card pe rand (nu 2/rand); tabelele dense read-only raman cu scroll contained |
|
||||
| `< 768px` | mobil | un card pe rand, o coloana, tinte touch 44px |
|
||||
|
||||
CSS custom properties NU functioneaza in `@media`; pragul se scrie literal
|
||||
(`@media (max-width:767px)`, `@media (max-width:1024px)`). Reutilizeaza aceste praguri,
|
||||
nu introduce altele noi.
|
||||
|
||||
---
|
||||
|
||||
## 4. Scara de control (tinte de atins)
|
||||
|
||||
| Context | min-height | Uz |
|
||||
|---------|-----------|-----|
|
||||
| Touch (`< 768px`) | **44px** | orice buton/link/select interactiv |
|
||||
| Desktop standard | **36px** | butoane, icon-btn, cardlink, intrari meniu |
|
||||
| Compact (desktop) | **32px** | kebab summary, butoane pager, pill-cat |
|
||||
|
||||
Pe desktop nu fortam 44px peste tot (devine greoi); pe mobil da. Latimea butoanelor:
|
||||
**auto, nu full-width**, cu exceptia butonului primar de actiune dintr-o bara dedicata
|
||||
(ex. „Trimite la RAR" in bara sticky, „Salveaza si continua").
|
||||
|
||||
---
|
||||
|
||||
## 5. Componente
|
||||
|
||||
### 5.1 Butoane — sistem unificat
|
||||
|
||||
Patru variante. Toate: `font:inherit` (IBM Plex Sans), `font-weight:500`, `border-radius:6px`,
|
||||
`padding:8px 14px` (desktop), tinta conform scarii de control. Tranzitie `filter/background .15s`.
|
||||
|
||||
| Varianta | Clasa | Fundal | Text | Bordura | Uz |
|
||||
|----------|-------|--------|------|---------|-----|
|
||||
| Primar | `.btn` (default `<button>`) | `--accent` | `#fff` | `--accent` | actiunea dominanta |
|
||||
| Secundar | `.btn-secondary` | transparent | `--ink` | `--line` | actiuni neutre (Editeaza, Filtreaza) |
|
||||
| Ghost | `.btn-ghost` | transparent | `--accent` | transparent | actiuni tertiare/linkuri-actiune |
|
||||
| Distructiv | `.btn-danger` | transparent | `--err` | `--err` | Sterge; hover → fundal `--err`, text `#fff` |
|
||||
|
||||
**Iconite in butoane:** label text + iconita optionala la stanga (16px, `fill:currentColor`,
|
||||
`aria-hidden`). **Butoanele icon-only sunt interzise pentru actiuni cu text echivalent**
|
||||
(ex. Salveaza/Sterge in tabele) — au cauzat „bloc colorat cu iconita invizibila" pe mobil.
|
||||
Cand spatiul e strans, foloseste un grup compact `[ Salveaza ] [ Sterge ]` cu text scurt,
|
||||
nu doua blocuri full-width unul sub altul. Icon-only ramane permis DOAR pentru: comutator
|
||||
tema, hamburger cont, kebab, inchidere modal — toate cu `aria-label`.
|
||||
|
||||
Stari: `:hover` → `filter:brightness(1.08)` (primar) sau `background:var(--line)` (secundar/ghost);
|
||||
`:focus-visible` → `outline:2px solid var(--accent); outline-offset:2px`; `:disabled` →
|
||||
`opacity:.45; cursor:default`. Stare „dirty" (modificari nesalvate) pe butonul de salvare:
|
||||
fundal `--accent`.
|
||||
|
||||
### 5.2 Card
|
||||
|
||||
`background:var(--card); border:1px solid var(--line); border-radius:10px`. Carduri de
|
||||
sectiune cu titlu `h2.sec` (15/600). Carduri-rand (lista pe mobil/tableta) cu radius 8–10px,
|
||||
stivuite vertical, gap intern 7–8px.
|
||||
|
||||
### 5.3 Tabel → card-uri (responsive)
|
||||
|
||||
Tabelele **actionabile** (Trimiteri, Preview import, Mapari) devin card-uri sub 1024px.
|
||||
Regula de card (vezi §3): NU folosi pattern-ul „eticheta cu `min-width` fix + valoare in
|
||||
flex" — sparge valorile pe verticala. In schimb:
|
||||
|
||||
- **Stivuieste**: eticheta mica deasupra valorii (`display:block`), SAU
|
||||
- **Card semantic**: linie titlu (identificator + stare), linii secundare mici. Preferat
|
||||
pentru liste lungi (Trimiteri, Preview).
|
||||
|
||||
Listele actionabile (Trimiteri, Preview, Mapari) raman **O COLOANA (un card pe rand)** pe
|
||||
TOT intervalul sub 1024px — nu se foloseste grila 2/rand. Decizie confirmata cu userul
|
||||
(gate 2026-06-27): simplitate si consecventa primeaza fata de densitate pe tableta.
|
||||
|
||||
Tabelele **dense read-only** (Jurnal, Nomenclator, Admin) raman tabel cu scroll orizontal
|
||||
**contained in card** (`.tablewrap { overflow-x:auto }`), nu se cardifica.
|
||||
|
||||
### 5.4 Stepper / wizard import — COMPACT pe toate latimile
|
||||
|
||||
Patru pasi: Incarca fisier · Potriveste coloanele · Verifica · Confirma trimiterea.
|
||||
|
||||
- **Desktop**: o bara slim orizontala — pastila numar (sau ✓) + titlu scurt pe O linie,
|
||||
pasul activ evidentiat. **Fara paragraf de ajutor inalt** in bara (ajutorul, daca e
|
||||
nevoie, e text mic sub bara, o singura linie). Inaltime tinta ~44px, nu blocuri inalte.
|
||||
- **Tableta/mobil**: colapsat la o singura linie — `Pasul N din 4 · <Titlu>` + bara de
|
||||
progres (`height:5px; border-radius:99px`, umplere `--accent` la `N/4`). Ajutorul pasului
|
||||
activ sub bara, text 12px muted.
|
||||
|
||||
Niciodata 4 coloane egale cu text — se taie/se rupe pe ecrane inguste.
|
||||
|
||||
### 5.5 Pill-uri si badge-uri
|
||||
|
||||
Stare: `.pill` 12px, radius 99px, cu clasa de culoare (`.s-ok/.s-warn/...`). Filtre de
|
||||
stare: `.pill-cat` (contur inactiv, umplere activa pe culoarea categoriei). Badge contor:
|
||||
cerc 18px, `--err`, text alb 11/700.
|
||||
|
||||
### 5.6 Formulare
|
||||
|
||||
Label 12–13px muted deasupra controlului. Input/select: `--bg`, bordura `--line`, radius 6px,
|
||||
padding `7px 10px`. Pe mobil controalele de formular din sectiunile de continut sunt
|
||||
full-width (tinta 44px). Pe desktop pastreaza latimi rezonabile (`select` max ~340px).
|
||||
|
||||
### 5.7 Modal
|
||||
|
||||
Desktop: dialog centrat `max-width:680px`, radius 12px, backdrop `rgba(0,0,0,.55)`,
|
||||
scroll intern. Mobil (`< 768px`): full-screen (fara colturi/umbra), buton inchidere 44px,
|
||||
focus-trap + scroll-lock + `inert` pe `<main>`.
|
||||
|
||||
---
|
||||
|
||||
## 6. Accesibilitate (obligatoriu)
|
||||
|
||||
- Contrast text >= 4.5:1 (normal), >= 3:1 (>=18px bold) in toate cele 3 palete.
|
||||
- Stare comunicata prin text, nu doar culoare.
|
||||
- `:focus-visible` vizibil pe tot ce e interactiv (outline `--accent`).
|
||||
- Tinte touch >= 44px pe mobil.
|
||||
- Icon-only obligatoriu cu `aria-label`; SVG decorativ `aria-hidden="true"`.
|
||||
- Modale: `role="dialog"`, `aria-modal`, focus-trap, focus return pe trigger.
|
||||
- `prefers-reduced-motion`: scurteaza/elimina tranzitiile non-esentiale.
|
||||
|
||||
---
|
||||
|
||||
## 7. Pentru planurile de design (cum se foloseste acest fisier)
|
||||
|
||||
Inainte de orice propunere vizuala:
|
||||
1. Citeste acest fisier integral. Foloseste DOAR tokenii de aici (culoare, type, radius, spatiu).
|
||||
2. Verifica fiecare ecran la 360 / 768 / 1024 / 1280px.
|
||||
3. Aplica compactarile globale (wizard, butoane) si pe desktop, nu doar pe mobil.
|
||||
4. Respecta „un singur primar per context" si scara de control.
|
||||
5. Daca o propunere cere un token nou (culoare/marime/radius), justifica si adauga-l AICI
|
||||
in acelasi PR — nu introduce valori ad-hoc in template.
|
||||
|
||||
Stadiul de implementare a regulilor responsive se urmareste in PRD-ul activ
|
||||
(`docs/prd/prd-5.13-responsive-compact.md`) si in `docs/ROADMAP.md`.
|
||||
@@ -1,277 +0,0 @@
|
||||
# Design 5.5 — Uniformizare & standardizare UI/UX
|
||||
|
||||
**Stare**: aprobat (decizii utilizator 2026-06-23, vezi §10)
|
||||
**Context**: dashboard web HTMX (`app/web/templates/`), paleta dark/light deja livrata (5.3),
|
||||
erori 3-niveluri (5.4). Acest document = sursa de adevar **vizuala** pentru PRD 5.5. Unde PRD-ul
|
||||
descrie *ce* livram pe stories, aici descriem *cum arata* si *de ce*.
|
||||
|
||||
> Nu reinventam estetica. Paleta, tipografia si tokenii din `base.html` (5.3) raman **NESCHIMBATI
|
||||
> la octet**. Standardizarea = aducem toate tabelele si paginile la acelasi vocabular de componente
|
||||
> care exista deja in tabelul Trimiteri (`_submissions.html`), tabelul considerat corect de referinta.
|
||||
|
||||
---
|
||||
|
||||
## 1. Problema (audit pe codul real)
|
||||
|
||||
Inventar al neuniformitatii curente:
|
||||
|
||||
| Suprafata | Simptom | Referinta corecta |
|
||||
|-----------|---------|-------------------|
|
||||
| Tabel **Mapari** (`_mapari.html`) | Labartat: coloana "Punere in coada" injecteaza prin macro `autosend_toggle` 3 randuri de text explicativ pe **fiecare** linie → randuri inalte, butoanele **Salveaza/Sterge** ies din viewport, trebuie scroll orizontal | grila Trimiteri |
|
||||
| Tabel **Nomenclator** (`_nomenclator.html`) | Functional dar minim; nu imparte exact acelasi aspect/hover/aliniere cu Trimiteri | grila Trimiteri |
|
||||
| **Acasa** (`_acasa.html`) | Sectiune "Ajutor: Mapari / Coduri RAR" redundanta (wayfinding repetat) | — (se elimina) |
|
||||
| **Navigare** | Cont, Integrare, Nomenclator stau ca tab-uri amestecate cu lucrul zilnic; logout + link admin sunt agatate ad-hoc in coltul dreapta-sus al dashboard-ului, absente pe alte pagini | meniu de cont dedicat |
|
||||
| **Panou admin** (`admin.html`) | Conturile in asteptare au doar "Activeaza" per-rand; lipsesc selectie multipla si actiunile blocare/arhivare/stergere. Nota "Cont dev implicit" e jargon intern nederivabil | tabel cu selectie + bara bulk |
|
||||
|
||||
Principiu de standardizare: **un singur tabel, o singura componenta de antet de sectiune, un singur
|
||||
loc pentru ajutor** (link/disclosure, nu text inline repetat pe randuri).
|
||||
|
||||
---
|
||||
|
||||
## 2. Design tokens (existenti — se reutilizeaza, nu se modifica)
|
||||
|
||||
Din `base.html` (`:root` dark + `[data-theme="light"]`). Citat aici doar ca referinta; **nicio
|
||||
valoare noua de culoare**. Orice suprafata noua foloseste `color-mix(... var(--card))` pentru stari
|
||||
(lectia 5.3: zero literali hardcodati, altfel se sparge light mode).
|
||||
|
||||
```
|
||||
--bg --card --ink --muted --line
|
||||
--ok (verde) --warn (chihlimbar) --err (rosu) --accent (albastru)
|
||||
```
|
||||
|
||||
Spacing: cardurile 16-20px padding; celule tabel `8px 10px`; gap-uri 6/8/12/16px (scara existenta).
|
||||
Radius: 6px controale, 10px carduri, 99px pill-uri. Tipografie: tabel 14px `tabular-nums`,
|
||||
antet `th` 12px uppercase `--muted`. **Nu introducem fonturi sau marimi noi.**
|
||||
|
||||
---
|
||||
|
||||
## 3. Componenta canonica: Tabelul standard
|
||||
|
||||
Tabelul Trimiteri defineste contractul. Orice tabel din aplicatie il respecta:
|
||||
|
||||
```
|
||||
.tablewrap > table
|
||||
thead th -> 12px uppercase, color --muted, font-weight 500, white-space nowrap
|
||||
tbody td -> 14px, padding 8px 10px, border-bottom 1px var(--line), nowrap implicit
|
||||
stare -> <span class="pill {s-*}">{text uman}</span> (glifa+text, nu doar culoare)
|
||||
coloana lunga (motiv) -> white-space:normal; max-width:280px (singura exceptie de la nowrap)
|
||||
empty state -> .empty (centrat, --muted, cu CTA contextual)
|
||||
```
|
||||
|
||||
Reguli care fac diferenta vizibila fata de "labartat":
|
||||
1. **Coloanele de control sunt inguste si nowrap.** Niciun text explicativ in celule. Explicatiile
|
||||
traiesc o singura data, in antetul cardului (link "Ajutor") sau intr-un `<details>`.
|
||||
2. **Actiunile incap fara scroll orizontal.** Coloana "Actiuni" la dreapta, `white-space:nowrap`,
|
||||
butoane scurte. Pe ecrane inguste scroll-ul ramane IN card (`.tablewrap`), nu in pagina.
|
||||
3. **Densitate constanta.** Inaltimea randului = o linie de text + padding. Sub-text (ex. "2 blocate",
|
||||
"acum: COD") merge in `<div class="muted" style="font-size:12px">` sub valoarea principala, nu
|
||||
pe coloana separata.
|
||||
|
||||
### 3.1 Antet de sectiune standard (cu Ajutor)
|
||||
|
||||
```
|
||||
+--------------------------------------------------------------+
|
||||
| De rezolvat [ Ajutor ] | <- h2 15px la stanga, link la dreapta
|
||||
+--------------------------------------------------------------+
|
||||
```
|
||||
|
||||
`Ajutor` = link discret `.cardlink` care comuta un `<details>`/panou de text (vezi §5). Mutam acolo
|
||||
toata proza care azi se repeta pe randuri. Un singur loc, citit la nevoie.
|
||||
|
||||
---
|
||||
|
||||
## 4. Tabelul Mapari — inainte / dupa
|
||||
|
||||
### Inainte (labartat)
|
||||
Fiecare rand din "De rezolvat" si "Mapari salvate" poarta `autosend_toggle`, care randeaza:
|
||||
- "La fisierele viitoare cu aceasta operatie:" (12px)
|
||||
- checkbox + **"Pune automat in coada"**
|
||||
- "Nebifat = «Tine pentru verificare». Doar pentru aceasta operatie; nimic nu pleaca la RAR..." (11px)
|
||||
|
||||
x N randuri. Coloana e mai lata decat selectul de cod; Salveaza/Sterge sunt impinse afara.
|
||||
|
||||
### Dupa (compact, ca Trimiteri)
|
||||
|
||||
```
|
||||
De rezolvat [ Ajutor ]
|
||||
-----------------------------------------------------------------
|
||||
OPERATIE SUGESTII COD RAR IN COADA ACTIUNI
|
||||
RevTehBP A012 (88%) [ A012 v ] (o) Auto [Salveaza]
|
||||
2 blocate ( ) Manual
|
||||
-----------------------------------------------------------------
|
||||
```
|
||||
|
||||
- Coloana **IN COADA** = comutator scurt cu doua stari etichetate **Auto** / **Manual** (radio sau
|
||||
switch), fara nicio propozitie. Tooltip pe control: "Auto = pune automat in coada la fisierele
|
||||
viitoare cu aceasta operatie; Manual = tine pentru verificare."
|
||||
- Explicatia completa (de ce exista maparile, ce inseamna Auto vs Manual, ce e blocat) → in panoul
|
||||
**Ajutor** din antet, scris o singura data.
|
||||
- **Invariant backend pastrat**: controlul emite tot `name="auto_send" value="true"` cu semantica de
|
||||
prezenta (bifat→true, absent→false), exact ca azi. Zero atingere backend (lectia 5.3/3.6:
|
||||
reskin la nivel de macro, parserele `/mapari` si `/_import/.../mapare-operatie` raman valide).
|
||||
- "Mapari salvate" si "Formate de coloane" → aceeasi grila; sub-textul ("acum: COD — nume",
|
||||
"N coloane", maparea coloana→camp) ramane `muted` 12px sub valoare, nu pe coloane separate verbose.
|
||||
|
||||
Macro-ul `autosend_toggle` se rescrie compact (acelasi `name`/`form`/`checked`), deci se schimba
|
||||
intr-un singur loc si se propaga si in fluxul de import (mapcoloane) unde e refolosit.
|
||||
|
||||
---
|
||||
|
||||
## 5. Panoul Ajutor (mapari)
|
||||
|
||||
Un `<details>` nativ in antetul cardului "De rezolvat" (sau link care expandeaza acelasi `<details>`),
|
||||
inchis implicit, fara JS:
|
||||
|
||||
```
|
||||
Ajutor (v)
|
||||
Maparile leaga o operatie din softul tau (cod intern ROAAUTO) de un cod RAR oficial.
|
||||
- Operatii necunoscute raman blocate in needs_mapping si NU pleaca la RAR pana le mapezi.
|
||||
- Sugestiile (%) vin din potrivire fuzzy pe denumire — verifica-le inainte sa salvezi.
|
||||
- In coada: Auto = la urmatoarele fisiere cu aceasta operatie, randurile intra automat in coada.
|
||||
Manual = raman pentru verificare; nimic nu pleaca la RAR pana confirmi tu.
|
||||
- La schimbarea unui cod, submission-urile blocate pe acea operatie se re-rezolva automat.
|
||||
```
|
||||
|
||||
Avantaj: text scris o data, accesibil, fara cost de inaltime pe fiecare rand. `<details>` =
|
||||
accesibil din tastatura nativ, fara dependente.
|
||||
|
||||
---
|
||||
|
||||
## 6. Navigare: meniu hamburger (decizie: dropdown ancorat dreapta-sus)
|
||||
|
||||
### 6.1 Header
|
||||
|
||||
```
|
||||
[Gateway RAR AUTOPASS] [test] [☀] v1.0 [☰]
|
||||
|
|
||||
+------------------+
|
||||
| Cont |
|
||||
| Integrare |
|
||||
| Nomenclator |
|
||||
| Panou admin | <- doar admin
|
||||
|------------------|
|
||||
| Iesi din cont | <- form POST /logout
|
||||
+------------------+
|
||||
```
|
||||
|
||||
- Iconita `☰` (`min 36x36`, `aria-label="Meniu cont"`, `aria-expanded`, `aria-controls`) langa toggle-ul
|
||||
de tema. Dropdown ancorat sub iconita, aliniat la dreapta. **Fara overlay** pe pagina.
|
||||
- Inchidere: click in afara, `Esc`, sau selectarea unui element. Focus trap minimal: `Esc` readuce
|
||||
focusul pe `☰`. Navigare cu sageti optionala (consistent cu pattern-ul tab existent), dar `Tab`
|
||||
natural e suficient.
|
||||
- Continut **dependent de autentificare** (vezi §6.3).
|
||||
|
||||
### 6.2 Tab-bar dupa mutare (decizie: doar Acasa · Mapari)
|
||||
|
||||
```
|
||||
[ Acasa ] [ Mapari ]
|
||||
```
|
||||
|
||||
Cont, Integrare, Nomenclator parasesc tab-bar-ul → meniul `☰`. Raman doar cele doua suprafete de
|
||||
**lucru zilnic**. Badge-urile de contoare (Mapari) raman pe tab. Deep-link `?tab=` si rutele
|
||||
`/_fragments/{cont,integrare,nomenclator}` raman valide (accesate acum din meniu, nu din tab-bar) —
|
||||
deci zero rute moarte, doar punctul de intrare se muta.
|
||||
|
||||
### 6.3 Stare de autentificare in header
|
||||
|
||||
`base.html` e partajat de `login.html`, `signup.html`, `dashboard.html`, `admin.html`. Meniul trebuie
|
||||
sa stie daca esti logat:
|
||||
|
||||
- **Autentificat**: arata Cont, Integrare, Nomenclator, (Panou admin daca `is_admin`), separator,
|
||||
"Iesi din cont" (form `POST /logout` cu `csrf_token`).
|
||||
- **Neautentificat** (login/signup): meniul arata doar "Autentificare" / "Inregistrare" (sau iconita
|
||||
`☰` ascunsa pe aceste pagini — vezi PRD US). Niciun link de cont, niciun logout.
|
||||
|
||||
Necesita ca `base.html` sa primeasca `is_authenticated`, `is_admin`, `csrf_token` in context. Se
|
||||
adauga ca un helper de context partajat (un singur loc), nu duplicat in fiecare render. Acesta e
|
||||
singurul "backend touch" din zona de navigare si trebuie sa fie aditiv si defensiv (lipsa cheilor →
|
||||
meniu in stare neautentificata, nu eroare).
|
||||
|
||||
---
|
||||
|
||||
## 7. Panou admin: selectie + actiuni bulk
|
||||
|
||||
### 7.1 Tabel conturi in asteptare (si analog conturi active)
|
||||
|
||||
```
|
||||
[v] Selecteaza tot 2 selectate:
|
||||
[Activeaza] [Blocheaza] [Arhiveaza] [Sterge] <- bara bulk, apare la selectie
|
||||
-----------------------------------------------------------------------
|
||||
[v] ID COMPANIE CUI EMAIL INREGISTRAT ACTIUNI
|
||||
[v] 7 Auto SRL RO123 a@b.ro 12.06.2026 [ ... ]
|
||||
[ ] 8 Moto SA RO456 c@d.ro 13.06.2026 [ ... ]
|
||||
-----------------------------------------------------------------------
|
||||
```
|
||||
|
||||
- Coloana de **checkbox** la stanga + un master "selecteaza tot" in antet.
|
||||
- **Bara de actiuni bulk** ascunsa pana exista o selectie; afiseaza numarul selectat si butoanele
|
||||
contextuale. Actioneaza pe toate randurile bifate (POST cu lista de `account_id`).
|
||||
- **Actiuni per-rand** in meniul `[ ... ]` (kebab): aceleasi verbe, pentru o singura tinta.
|
||||
- Verbele, ca stari de cont distincte (vezi §7.2): **Activeaza, Blocheaza, Arhiveaza, Sterge**.
|
||||
- `Sterge` = actiune distructiva → `hx-confirm` / dialog de confirmare obligatoriu.
|
||||
- `Blocheaza`/`Arhiveaza` reversibile → confirmare doar pe bulk (cantitate).
|
||||
- Stari vizuale ale verbelor: distructiv (`Sterge`) cu `color:var(--err)`; restul neutre `.cardlink`.
|
||||
|
||||
### 7.2 Model de stare cont (impact backend — vezi PRD riscuri)
|
||||
|
||||
Azi: `accounts.active` (0/1); "pending" = inregistrat dar `active=0`. Cele 4 verbe cer stari
|
||||
distincte care nu incap intr-un bool. Propunere (PRD o ratifica): coloana `accounts.status`
|
||||
TEXT, migrare defensiva, derivata din `active` la prima rulare:
|
||||
|
||||
```
|
||||
pending -> inregistrat, neactivat inca (active=0, status nesetat istoric)
|
||||
active -> operational (active=1)
|
||||
blocked -> suspendat reversibil (nu logheaza, worker nu trimite)
|
||||
archived -> ascuns din liste, date pastrate (read-only)
|
||||
deleted -> stergere (GDPR/L.142) (hard delete SAU soft cu purge)
|
||||
```
|
||||
|
||||
- Worker `claim_one` gate-uieste pe **status='active'** (azi pe `COALESCE(active,1)=1`) — schimbare
|
||||
semantica de pastrat compatibila: `active=1 ⇔ status='active'`.
|
||||
- **Contul dev id=1 e protejat** de Blocheaza/Arhiveaza/Sterge (cont de sistem), exact ca azi la
|
||||
activate/deactivate. Daca e selectat in bulk, e sarit, nu eroare.
|
||||
- Nota "Cont dev implicit (id=1)" din pagina **se elimina** (jargon intern, nederivabil de operator).
|
||||
Protectia ramane in cod, nu o explicam in UI.
|
||||
|
||||
> Aceasta e singura zona cu schema/backend real. Restul livrabilei e UI pur (reskin + reasezare).
|
||||
|
||||
---
|
||||
|
||||
## 8. Accesibilitate & paritate tema
|
||||
|
||||
- **AA pe light+dark** pentru orice text nou (lectia 5.3: verzi/rosii hardcodate cad sub AA). Stari
|
||||
doar prin `color-mix(... var(--card))`, niciun literal.
|
||||
- Stare = **glifa + text**, nu doar culoare (pill-urile existente respecta deja).
|
||||
- Meniul `☰`: `aria-expanded`, `aria-controls`, inchidere pe `Esc`, focus readus pe trigger.
|
||||
- `<details>` Ajutor: accesibil nativ din tastatura.
|
||||
- Checkbox-uri admin: `aria-label` per rand ("Selecteaza contul {companie}"); master = "Selecteaza tot".
|
||||
- Toate controalele >=36px zona de atins (consistent cu toggle tema / `.cardlink`).
|
||||
|
||||
## 9. Motiune
|
||||
|
||||
Minima, consistenta cu existentul: dropdown `☰` fade/translate scurt (~120ms, ca `.tab-link`
|
||||
transition). `<details>` = comportament nativ. Fara animatii noi de amploare. Respecta
|
||||
`prefers-reduced-motion` daca adaugam tranzitii (omitere la cerere).
|
||||
|
||||
## 10. Decizii utilizator (2026-06-23)
|
||||
|
||||
1. Meniu hamburger = **dropdown ancorat dreapta-sus** (nu drawer).
|
||||
2. Tab-bar = **Acasa · Mapari**; Nomenclator + Cont + Integrare + Panou admin → meniul `☰`.
|
||||
3. Mapari = **grila compacta ca Trimiteri**, toggle scurt **Auto/Manual**, link **Ajutor** in antet;
|
||||
textul repetat de pe randuri se elimina.
|
||||
4. Admin = **selectie cu bife + bara de actiuni bulk** (Activeaza/Blocheaza/Arhiveaza/Sterge) +
|
||||
actiuni per-rand; nota "cont dev implicit" **eliminata**.
|
||||
5. Sectiunea "Ajutor" de pe Acasa se **elimina**.
|
||||
6. Nomenclator capata exact aspectul tabelului Trimiteri.
|
||||
|
||||
## 11. Componente atinse (harta pentru PRD)
|
||||
|
||||
| Componenta | Fisier | Tip schimbare |
|
||||
|------------|--------|---------------|
|
||||
| Header + meniu `☰` + context auth | `base.html` (+ context render rute) | reasezare + mic backend context |
|
||||
| Tab-bar redus | `dashboard.html` | reasezare |
|
||||
| Acasa fara Ajutor | `_acasa.html` | stergere |
|
||||
| Mapari standard + toggle compact + Ajutor | `_mapari.html`, `_macros.html` | reskin UI (zero backend) |
|
||||
| Nomenclator ca Trimiteri | `_nomenclator.html` | reskin UI |
|
||||
| Admin selectie + bulk + verbe noi | `admin.html` + rute `/admin/*` | UI + **backend (status)** |
|
||||
| Model stare cont | `schema.sql`, `users.py`, worker gate | **backend + migrare** |
|
||||
112
docs/landing-page-prompt.md
Normal file
112
docs/landing-page-prompt.md
Normal file
@@ -0,0 +1,112 @@
|
||||
# Prompt landing page — claude.ai/design
|
||||
|
||||
Referinta pentru generarea unui mockup de landing page comerciala (homepage public) a
|
||||
produsului. De copiat ca atare in claude.ai/design. Tokenii de culoare/tipografie/control sunt
|
||||
preluati din `design.md` (sursa de adevar a sistemului de design) — orice modificare la paleta
|
||||
sau scara de control se face intai acolo, apoi se reflecta aici.
|
||||
|
||||
---
|
||||
|
||||
```
|
||||
Creează un mockup pentru o LANDING PAGE comercială (homepage public) a unui produs SaaS B2B românesc.
|
||||
|
||||
== PRODUS ==
|
||||
Nume: Gateway RAR AUTOPASS (de la ROMFAST).
|
||||
Ce face: un serviciu web prin care atelierele service-auto din România își declară automat
|
||||
prestațiile la sistemul RAR AUTOPASS — o obligație legală (Legea 142/2023, OM 210/2024).
|
||||
Înlocuiește raportarea manuală. Publicul țintă: administratori și recepționeri de service auto,
|
||||
oameni ne-tehnici, ocupați. Mesajul cheie: "Conformitate RAR fără bătaie de cap — încarci un
|
||||
fișier sau conectezi softul de service, noi trimitem la RAR în siguranță."
|
||||
|
||||
Tonul: serios, de încredere, instituțional dar modern. NU startup glumeț, NU emoji.
|
||||
Limba: română (cu diacritice în text vizibil). Fără emoji nicăieri.
|
||||
|
||||
== OBIECTIV PAGINĂ ==
|
||||
Convinge un service auto că produsul rezolvă obligația legală RAR simplu și sigur, și să se
|
||||
înregistreze. Acțiune dominantă: "Creează cont" / "Autentificare".
|
||||
Cârlig comercial principal: GRATUIT pentru service-urile mici, până la 100 de prezentări pe lună.
|
||||
|
||||
== SISTEM DE CULORI (folosește exact aceste valori — temă întunecată ca implicit) ==
|
||||
Fundal pagină: #0f1218
|
||||
Suprafață card: #181c24
|
||||
Text principal: #e6e9ef
|
||||
Text secundar: #8b93a7
|
||||
Borduri/linii: #262b36
|
||||
Accent (CTA/link): #2E74D6 (azur ROMFAST)
|
||||
Succes: #2FBF8F
|
||||
Atenție: #E0A93B
|
||||
Eroare: #E05D5D
|
||||
Oferă și o variantă pe temă LIGHT: fundal #f5f7fa, card #ffffff, text #1a1d24,
|
||||
text secundar #5c6473, linii #e2e5ea, accent #1F66C9.
|
||||
|
||||
== TIPOGRAFIE ==
|
||||
Font UI: IBM Plex Sans (weights 400/500/700).
|
||||
Font mono (pentru coduri, VIN, ID-uri tehnice dacă apar): IBM Plex Mono.
|
||||
Titlu hero mare și greu (700, letter-spacing ușor negativ). Corp 15–16px, secundar 12–13px.
|
||||
|
||||
== STIL CONTROALE ==
|
||||
Butoane: radius 6px, padding ~10px 18px, înălțime ~44px.
|
||||
- Primar: fundal #2E74D6, text alb, hover ușor mai luminos.
|
||||
- Secundar: transparent, text deschis, bordură #262b36.
|
||||
Carduri: fundal #181c24, bordură 1px #262b36, radius 10px, plat (umbră doar la elemente plutitoare).
|
||||
Pill/badge: radius 99px. Iconițe simple liniare, monocrome, NU ilustrații colorate gen 3D.
|
||||
Aspect plat, sobru, mult spațiu de respirație. Densitate compactă dar aerisită.
|
||||
|
||||
== STRUCTURĂ PAGINĂ (de sus în jos) ==
|
||||
1. HEADER fix: stânga logo "ROMFAST" (text wordmark, accent azur pe "FAST"); dreapta două
|
||||
butoane: "Autentificare" (secundar) + "Creează cont" (primar). Comutator temă opțional.
|
||||
|
||||
2. HERO: titlu puternic (ex. "Declară prestațiile la RAR AUTOPASS, automat"), subtitlu de o frază
|
||||
despre conformitate legală fără efort. Sub titlu, un badge/pill verde vizibil:
|
||||
"Gratuit până la 100 de prezentări pe lună". Apoi un buton primar mare "Creează cont gratuit"
|
||||
+ un buton secundar "Vezi cum funcționează". Sub butoane, o linie mică de încredere
|
||||
(ex. "Conform Legii 142/2023 · datele tale criptate · fără card bancar"). În dreapta hero:
|
||||
un mockup vizual al dashboardului aplicației (un card cu o listă de "Trimiteri" cu pill-uri de
|
||||
stare colorate: Trimis=verde, În coadă=albastru, Eroare=roșu).
|
||||
|
||||
3. BANDĂ DE ÎNCREDERE: "Construit de ROMFAST" + mențiune că înlocuiește integrarea/raportarea manuală.
|
||||
|
||||
4. SECȚIUNE "Cum funcționează" — 3 pași cu iconițe liniare:
|
||||
(1) Încarci fișierul (xlsx/csv) sau conectezi softul de service prin API,
|
||||
(2) Verifici și mapezi coloanele o singură dată (le ținem minte pentru data viitoare),
|
||||
(3) Trimitem automat la RAR, tu urmărești starea live.
|
||||
|
||||
5. SECȚIUNE BENEFICII — de ce merită această interfață (6 carduri scurte, fiecare cu iconiță liniară):
|
||||
- "Zero raportare manuală" — încarci un fișier, gata; nu mai introduci prezentări una câte una în portalul RAR.
|
||||
- "Mapare reținută" — potrivești coloanele o singură dată per format; fișierele următoare se completează singure.
|
||||
- "Anti-duplicat" — verificare automată ca aceeași prezentare să nu ajungă de două ori la RAR (RAR nu permite anulare).
|
||||
- "Validare înainte de trimitere" — erorile (VIN, cod prestație, kilometraj) sunt prinse și explicate înainte să meargă la RAR.
|
||||
- "Date criptate (GDPR)" — datele cu caracter personal sunt criptate și șterse automat după perioada legală.
|
||||
- "Stare live" — vezi în timp real ce s-a trimis, ce e în coadă și ce trebuie corectat.
|
||||
|
||||
6. SECȚIUNE INTEGRARE API (pentru service-uri cu soft propriu / ROAAUTO):
|
||||
Card mai mare, pe două coloane. Stânga: titlu "Ai deja un soft de service? Conectează-l direct."
|
||||
text scurt despre faptul că nu mai e nevoie de export manual — softul tău trimite prezentările
|
||||
automat printr-un singur apel API, cu cheie API per cont. Dreapta: un mic bloc de cod cu font
|
||||
mono (IBM Plex Mono) care arată un exemplu de request, ex.:
|
||||
POST /v1/prezentari
|
||||
Authorization: rfak_••••••••
|
||||
{ "vin": "...", "cod_prestatie": "...", "odometru": ... }
|
||||
Sub el un link/buton secundar "Vezi documentația API".
|
||||
|
||||
7. SECȚIUNE PREȚ — simplă, 2 planuri unul lângă altul:
|
||||
- "Gratuit": "0 lei/lună · până la 100 de prezentări/lună · import web + API · toate funcțiile de bază".
|
||||
Marcat ca recomandat pentru service-uri mici. Buton primar "Începe gratuit".
|
||||
- "Volum mare": "Pentru service-uri cu peste 100 de prezentări/lună · contactează-ne".
|
||||
Buton secundar "Contact".
|
||||
Accent pe ideea că pentru un service mic nu costă nimic, fără card bancar la înscriere.
|
||||
|
||||
8. CTA FINAL: card mare centrat "Începe să declari la RAR în câteva minute" + buton primar "Creează cont gratuit".
|
||||
|
||||
9. FOOTER: linkuri (Termeni, Confidențialitate/GDPR, Documentație API, Contact), © ROMFAST, mențiune legală.
|
||||
|
||||
== RESPONSIVE ==
|
||||
Arată DOUĂ artboard-uri: desktop (1280px) și mobil (375px). Pe mobil: o coloană, butoanele CTA
|
||||
full-width, header colapsat, cardurile de preț stivuite vertical. Ținte de atins minim 44px pe mobil.
|
||||
|
||||
== ACCESIBILITATE ==
|
||||
Contrast text minim 4.5:1. Starea comunicată prin text + culoare (nu doar culoare) — pill-urile de
|
||||
stare au etichetă text, nu doar pastilă colorată.
|
||||
|
||||
Livrează mockup-ul în tema întunecată ca varianta principală, plus un preview al heroului pe tema light.
|
||||
```
|
||||
194
docs/mockups/prd-5.15-mockups.html
Normal file
194
docs/mockups/prd-5.15-mockups.html
Normal file
@@ -0,0 +1,194 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>PRD 5.15 — mockup-uri piese lipsa (D6 strip / E4 picker / odo reveal)</title>
|
||||
<link rel="preconnect" href="https://fonts.googleapis.com">
|
||||
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
|
||||
<link href="https://fonts.googleapis.com/css2?family=IBM+Plex+Mono:wght@400;500&family=IBM+Plex+Sans:wght@400;500;700&display=swap" rel="stylesheet">
|
||||
<style>
|
||||
*{box-sizing:border-box;}
|
||||
/* grafit (default dark) — aceleasi tokenuri ca landing.html */
|
||||
:root{--bg:#0f1218;--card:#181c24;--card2:#0f1218;--text:#e6e9ef;--sub:#8b93a7;--line:#262b36;--line2:#1f2530;--accent:#2E74D6;--okt:#2FBF8F;--infot:#6ea2ec;--errt:#E05D5D;--warn:#E0A93B;}
|
||||
body{margin:0;padding:32px;background:#0b0e13;font-family:'IBM Plex Sans',system-ui,sans-serif;-webkit-font-smoothing:antialiased;color:var(--text);}
|
||||
h1{font:700 22px 'IBM Plex Sans';margin:0 0 4px;}
|
||||
.pgsub{font:400 13px 'IBM Plex Sans';color:var(--sub);margin:0 0 28px;}
|
||||
.seclabel{font:500 13px 'IBM Plex Sans';color:var(--sub);letter-spacing:.04em;text-transform:uppercase;margin:34px 0 14px;border-top:1px solid var(--line);padding-top:18px;}
|
||||
.frames{display:flex;gap:24px;flex-wrap:wrap;align-items:flex-start;}
|
||||
.frlabel{font:500 12px 'IBM Plex Sans';color:var(--sub);margin-bottom:10px;}
|
||||
/* componente slim */
|
||||
.counter{flex:1;background:var(--card2);border:1px solid var(--line);border-radius:8px;padding:10px 12px;}
|
||||
.cnum{font:700 22px 'IBM Plex Sans';line-height:1;}
|
||||
.clabel{font:400 11px 'IBM Plex Sans';color:var(--sub);margin-top:5px;}
|
||||
.csub{font:400 10px 'IBM Plex Mono';color:var(--sub);margin-top:3px;}
|
||||
.row{display:flex;align-items:center;justify-content:space-between;padding:11px 18px;border-bottom:1px solid var(--line2);}
|
||||
.vin{font:500 13px 'IBM Plex Mono';color:var(--text);}
|
||||
.meta{font:400 11px 'IBM Plex Sans';color:var(--sub);margin-top:3px;}
|
||||
.pill{display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;font:500 12px 'IBM Plex Sans';}
|
||||
.dot{width:6px;height:6px;border-radius:99px;}
|
||||
.lab{font:400 11px 'IBM Plex Sans';color:var(--sub);margin-bottom:4px;}
|
||||
.fld{height:30px;display:flex;align-items:center;padding:0 10px;border:1px solid var(--line);border-radius:6px;background:var(--card2);font:400 12px 'IBM Plex Mono';color:var(--text);}
|
||||
.fldsans{font-family:'IBM Plex Sans';}
|
||||
.chip{display:inline-flex;align-items:center;gap:5px;padding:3px 8px;border-radius:5px;background:color-mix(in srgb,var(--accent) 18%,transparent);color:var(--accent);font:600 11px 'IBM Plex Mono';}
|
||||
.chipx{opacity:.7;cursor:pointer;}
|
||||
.chipbox{min-height:30px;display:flex;align-items:center;gap:6px;flex-wrap:wrap;padding:4px 8px;border:1px solid var(--line);border-radius:6px;background:var(--card2);}
|
||||
.addcode{display:inline-flex;align-items:center;height:22px;padding:0 7px;border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line));border-radius:5px;color:var(--accent);font:500 10px 'IBM Plex Sans';cursor:pointer;}
|
||||
.oprow{display:flex;align-items:center;justify-content:space-between;gap:10px;padding:8px 10px;border:1px solid var(--line);border-radius:6px;background:var(--card2);margin-bottom:8px;}
|
||||
.opname{font:500 12px 'IBM Plex Sans';color:var(--text);}
|
||||
.picker{height:26px;display:inline-flex;align-items:center;gap:6px;padding:0 8px;border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line));border-radius:5px;background:transparent;color:var(--accent);font:500 11px 'IBM Plex Sans';cursor:pointer;}
|
||||
.saverule{font:400 10px 'IBM Plex Sans';color:var(--okt);margin-top:3px;display:inline-flex;align-items:center;gap:4px;}
|
||||
.btn{margin-top:6px;height:34px;padding:0 16px;border-radius:6px;background:var(--accent);border:none;color:#fff;font:600 12px 'IBM Plex Sans';cursor:pointer;align-self:flex-start;}
|
||||
.form{display:flex;flex-direction:column;gap:11px;padding:18px;}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<h1>PRD 5.15 — mockup-uri pentru piesele fara design</h1>
|
||||
<p class="pgsub">Tema grafit (dark), tokenuri identice cu landing.html. Trei piese pe care mockup-urile existente nu le acopera: stripul de sanatate D6, pickerul prestatii E4 (op↔cod), si reveal-ul odometru initial.</p>
|
||||
|
||||
<!-- ===================== D6 STRIP SANATATE ===================== -->
|
||||
<div class="seclabel">1 · D6 — strip sanatate mereu-vizibil (deasupra contoarelor)</div>
|
||||
<div class="frames">
|
||||
|
||||
<div>
|
||||
<div class="frlabel">Stare BLOCAT (rosu) — declaratiile NU pleaca</div>
|
||||
<div style="width:600px;background:var(--bg);border:1px solid var(--line);border-radius:12px;overflow:hidden;">
|
||||
<div style="padding:14px;">
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;gap:12px;padding:10px 14px;border-radius:8px;background:color-mix(in srgb,var(--errt) 16%,var(--card));border:1px solid color-mix(in srgb,var(--errt) 40%,transparent);margin-bottom:14px;">
|
||||
<div style="display:flex;align-items:center;gap:9px;">
|
||||
<span aria-hidden="true" style="font:700 15px 'IBM Plex Sans';color:var(--errt);">✗</span>
|
||||
<span style="font:700 13px 'IBM Plex Sans';color:var(--text);">Blocat: worker oprit — declaratiile NU pleaca</span>
|
||||
</div>
|
||||
<span style="font:400 11px 'IBM Plex Mono';color:var(--sub);white-space:nowrap;">Ultima autentificare RAR: azi 08:14</span>
|
||||
</div>
|
||||
<div style="margin-bottom:14px;">
|
||||
<div style="font:700 17px 'IBM Plex Sans';color:var(--text);">Trimiteri RAR AUTOPASS</div>
|
||||
<div style="font:400 12px 'IBM Plex Sans';color:var(--sub);margin-top:2px;">Service Auto Valcea · 28 iun 2026</div>
|
||||
</div>
|
||||
<div style="display:flex;gap:12px;margin-bottom:14px;">
|
||||
<div class="counter"><div class="cnum" style="color:var(--text);">847</div><div class="clabel">Trimise (total)</div><div class="csub">luna 124 · azi 9</div></div>
|
||||
<div class="counter"><div class="cnum" style="color:var(--accent);">12</div><div class="clabel">In coada</div></div>
|
||||
<div class="counter"><div class="cnum" style="color:var(--errt);">2</div><div class="clabel">De corectat</div></div>
|
||||
</div>
|
||||
<div>
|
||||
<div class="row"><div><div class="vin">WBA8E9...K7F2</div><div class="meta">Inspectie tehnica · 09:42</div></div><span class="pill" style="background:color-mix(in srgb,var(--okt) 13%,transparent);color:var(--okt);"><span class="dot" style="background:var(--okt);"></span>Trimis</span></div>
|
||||
<div class="row"><div><div class="vin">WVWZZZ...3M1</div><div class="meta">Revizie periodica · 09:38</div></div><span class="pill" style="background:color-mix(in srgb,var(--accent) 14%,transparent);color:var(--infot);"><span class="dot" style="background:var(--accent);"></span>In coada</span></div>
|
||||
<div class="row" style="border-bottom:none;"><div><div class="vin">VF1RFB...A88</div><div class="meta">Sistem franare · 09:31</div></div><span class="pill" style="background:color-mix(in srgb,var(--errt) 14%,transparent);color:var(--errt);"><span class="dot" style="background:var(--errt);"></span>De corectat</span></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<div class="frlabel">Stare OK (verde) — declaratiile curg</div>
|
||||
<div style="width:600px;background:var(--bg);border:1px solid var(--line);border-radius:12px;overflow:hidden;">
|
||||
<div style="padding:14px;">
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;gap:12px;padding:10px 14px;border-radius:8px;background:color-mix(in srgb,var(--okt) 13%,transparent);border:1px solid color-mix(in srgb,var(--okt) 30%,transparent);margin-bottom:14px;">
|
||||
<div style="display:flex;align-items:center;gap:9px;">
|
||||
<span aria-hidden="true" style="font:700 15px 'IBM Plex Sans';color:var(--okt);">✓</span>
|
||||
<span style="font:600 13px 'IBM Plex Sans';color:var(--text);">Declaratiile curg normal</span>
|
||||
</div>
|
||||
<span style="font:400 11px 'IBM Plex Mono';color:var(--sub);white-space:nowrap;">Ultima autentificare RAR: azi 09:12</span>
|
||||
</div>
|
||||
<div style="margin-bottom:14px;">
|
||||
<div style="font:700 17px 'IBM Plex Sans';color:var(--text);">Trimiteri RAR AUTOPASS</div>
|
||||
<div style="font:400 12px 'IBM Plex Sans';color:var(--sub);margin-top:2px;">Service Auto Valcea · 28 iun 2026</div>
|
||||
</div>
|
||||
<div style="display:flex;gap:12px;margin-bottom:14px;">
|
||||
<div class="counter"><div class="cnum" style="color:var(--text);">847</div><div class="clabel">Trimise (total)</div><div class="csub">luna 124 · azi 9</div></div>
|
||||
<div class="counter"><div class="cnum" style="color:var(--accent);">3</div><div class="clabel">In coada</div></div>
|
||||
<div class="counter"><div class="cnum" style="color:var(--sub);">0</div><div class="clabel">De corectat</div></div>
|
||||
</div>
|
||||
<div>
|
||||
<div class="row"><div><div class="vin">WBA8E9...K7F2</div><div class="meta">Inspectie tehnica · 09:42</div></div><span class="pill" style="background:color-mix(in srgb,var(--okt) 13%,transparent);color:var(--okt);"><span class="dot" style="background:var(--okt);"></span>Trimis</span></div>
|
||||
<div class="row" style="border-bottom:none;"><div><div class="vin">ZAR937...C04</div><div class="meta">Schimb ulei · 09:24</div></div><span class="pill" style="background:color-mix(in srgb,var(--okt) 13%,transparent);color:var(--okt);"><span class="dot" style="background:var(--okt);"></span>Trimis</span></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
<!-- ===================== E4 PICKER PRESTATII ===================== -->
|
||||
<div class="seclabel">2 · E4 — formular editare slim: VIN unic + Observatii + picker prestatii PE operatie</div>
|
||||
<div class="frames">
|
||||
<div>
|
||||
<div class="frlabel">Editare trimitere (needs_data)</div>
|
||||
<div style="width:640px;background:var(--card);border:1px solid var(--line);border-radius:10px;overflow:hidden;">
|
||||
<div class="form">
|
||||
<div style="display:flex;align-items:center;justify-content:space-between;margin-bottom:2px;">
|
||||
<span style="font:500 12px 'IBM Plex Mono';color:var(--sub);">corecteaza · needs_data</span>
|
||||
<span style="display:inline-flex;align-items:center;gap:6px;padding:4px 10px;border-radius:99px;background:color-mix(in srgb,var(--errt) 14%,transparent);color:var(--errt);font:600 11px 'IBM Plex Sans';">Date incomplete</span>
|
||||
</div>
|
||||
<div><div class="lab">VIN (serie sasiu)</div><div class="fld">U1234567890123456</div></div>
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:11px;">
|
||||
<div><div class="lab">Data prestatiei</div><div class="fld">2026-06-22</div></div>
|
||||
<div><div class="lab">Numar inmatriculare</div><div class="fld">CT88NOE</div></div>
|
||||
</div>
|
||||
<div><div class="lab">Observatii (operatiile efectuate)</div><div class="fld fldsans" style="height:auto;min-height:48px;align-items:flex-start;padding:8px 10px;">Revizie; schimbare placute frana</div></div>
|
||||
<div>
|
||||
<div class="lab">Prestatii — cod RAR pe fiecare operatie</div>
|
||||
<div class="oprow">
|
||||
<span class="opname">REVIZIE PERIODICA</span>
|
||||
<span style="display:flex;align-items:center;gap:8px;">
|
||||
<span class="chip"><span class="chipx">×</span>REV2</span>
|
||||
<span class="picker">+ alt cod</span>
|
||||
</span>
|
||||
</div>
|
||||
<div class="saverule">✓ salveaza regula REVIZIE PERIODICA → REV2 (auto-rezolva data viitoare)</div>
|
||||
<div class="oprow" style="border-color:color-mix(in srgb,var(--warn) 45%,var(--line));margin-top:10px;">
|
||||
<span class="opname">SCHIMBARE PLACUTE FRANA <span style="color:var(--warn);font:500 10px 'IBM Plex Sans';">· lipsa cod</span></span>
|
||||
<span class="picker" style="border-style:solid;border-color:var(--warn);color:var(--warn);">alege cod RAR ▾</span>
|
||||
</div>
|
||||
<div style="font:400 10px 'IBM Plex Sans';color:var(--sub);margin-top:8px;">Fara operatie (corectie pura): chip-uri de coduri libere · dedupare per-pereche (op,cod), nu doar dupa cod.</div>
|
||||
</div>
|
||||
<button class="btn">Salveaza si retrimite</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ===================== ODO REVEAL ===================== -->
|
||||
<div class="seclabel">3 · Reveal odometru initial — apare doar la coduri R-ODO / I-ODO (server-driven, E6)</div>
|
||||
<div class="frames">
|
||||
|
||||
<div>
|
||||
<div class="frlabel">Inainte · niciun R-ODO → odometru initial ascuns</div>
|
||||
<div style="width:480px;background:var(--card);border:1px solid var(--line);border-radius:10px;overflow:hidden;">
|
||||
<div class="form">
|
||||
<div><div class="lab">Prestatii</div>
|
||||
<div class="chipbox"><span class="chip"><span class="chipx">×</span>REV2</span><span class="addcode">+ cod</span></div>
|
||||
</div>
|
||||
<div><div class="lab">Odometru final</div><div class="fld">39000</div></div>
|
||||
<div style="font:400 10px 'IBM Plex Sans';color:var(--sub);font-style:italic;">Odometru initial se cere doar pentru coduri R-ODO / I-ODO.</div>
|
||||
<button class="btn">Salveaza</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<div class="frlabel">Dupa · adaugi R-ODO → campul apare</div>
|
||||
<div style="width:480px;background:var(--card);border:1px solid var(--line);border-radius:10px;overflow:hidden;">
|
||||
<div class="form">
|
||||
<div><div class="lab">Prestatii</div>
|
||||
<div class="chipbox">
|
||||
<span class="chip"><span class="chipx">×</span>REV2</span>
|
||||
<span class="chip" style="background:color-mix(in srgb,var(--warn) 22%,transparent);color:var(--warn);"><span class="chipx">×</span>R-ODO</span>
|
||||
<span class="addcode">+ cod</span>
|
||||
</div>
|
||||
</div>
|
||||
<div><div class="lab">Odometru final</div><div class="fld">39000</div></div>
|
||||
<div style="border-left:2px solid var(--warn);padding-left:10px;margin-left:-12px;">
|
||||
<div class="lab" style="color:var(--warn);">Odometru initial · necesar pentru R-ODO</div>
|
||||
<div class="fld" style="border-color:color-mix(in srgb,var(--warn) 50%,var(--line));">12500</div>
|
||||
</div>
|
||||
<button class="btn">Salveaza</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
</body>
|
||||
</html>
|
||||
221
docs/mockups/prd-5.16-dashboard-mobil.html
Normal file
221
docs/mockups/prd-5.16-dashboard-mobil.html
Normal file
@@ -0,0 +1,221 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>PRD 5.16 — Dashboard mobil 390px (RAR dot in antet + meniu)</title>
|
||||
<style>
|
||||
:root{
|
||||
--font-ui: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
||||
--font-mono: ui-monospace, "SF Mono", "Cascadia Code", "Segoe UI Mono", "Roboto Mono", Menlo, Consolas, monospace;
|
||||
--fs-xs:12px; --fs-sm:13.5px; --fs-base:15px; --fs-md:16px; --fs-lg:18px; --fs-xl:20px; --fs-2xl:28px;
|
||||
--bg:#0f1218; --card:#181c24; --card2:#0f1218; --ink:#e6e9ef; --muted:#8b93a7;
|
||||
--line:#262b36; --line2:#1f2530; --accent:#6ea2ec; --ok:#2FBF8F; --warn:#E0A93B; --err:#E05D5D;
|
||||
--hbg:rgba(15,18,24,.95);
|
||||
}
|
||||
*{box-sizing:border-box;}
|
||||
body{margin:0; background:#05070b; font-family:var(--font-ui); -webkit-font-smoothing:antialiased; padding:24px;}
|
||||
.mono{font-family:var(--font-mono);} .muted{color:var(--muted);}
|
||||
.stage{display:flex; gap:34px; justify-content:center; align-items:flex-start; flex-wrap:wrap;}
|
||||
.cap{text-align:center; color:#9aa3b2; font-size:13px; margin-top:10px; max-width:390px;}
|
||||
.phone{width:390px; background:var(--bg); color:var(--ink); border-radius:30px; border:10px solid #20242c; overflow:hidden; box-shadow:0 30px 70px -20px rgba(0,0,0,.7);}
|
||||
.phone .screen{height:720px; overflow:hidden; position:relative;}
|
||||
.scroll{height:100%; overflow:auto;}
|
||||
|
||||
header{position:sticky; top:0; z-index:5; display:flex; align-items:center; justify-content:space-between; gap:8px; height:56px; padding:0 12px; background:var(--hbg); backdrop-filter:blur(8px); border-bottom:1px solid var(--line);}
|
||||
.logo-fallback{display:inline-flex; align-items:center; gap:4px; font-weight:800; font-size:var(--fs-base);}
|
||||
.logo-fallback .rom{color:#D1342F;} .logo-fallback .fast{color:var(--accent);}
|
||||
.h-center{flex:1; text-align:center; line-height:1.1; min-width:0;}
|
||||
.h-title{font-size:var(--fs-sm); font-weight:700;} .h-title .accent{color:var(--accent);}
|
||||
.tier{display:inline-block; margin-left:5px; padding:0 7px; border-radius:99px; font-size:9px; font-weight:700; text-transform:uppercase; letter-spacing:.04em; color:var(--accent); background:color-mix(in srgb,var(--accent) 16%,transparent); vertical-align:middle;}
|
||||
.h-sub{font-size:11px; color:var(--muted); margin-top:1px; white-space:nowrap; overflow:hidden; text-overflow:ellipsis;}
|
||||
.h-sub .svc{color:var(--ink); font-weight:600;}
|
||||
.h-right{display:flex; align-items:center; gap:7px;}
|
||||
/* RAR online = dot compact in antet (title pe hover); blocat => rosu */
|
||||
.rar-dot{width:38px; height:38px; border-radius:9px; border:1px solid color-mix(in srgb,var(--ok) 35%,var(--line)); background:color-mix(in srgb,var(--ok) 10%,transparent); display:inline-flex; align-items:center; justify-content:center; cursor:default;}
|
||||
.rar-dot .d{width:11px; height:11px; border-radius:99px; background:var(--ok); box-shadow:0 0 0 4px color-mix(in srgb,var(--ok) 22%,transparent);}
|
||||
.icon-btn{width:40px; height:40px; border-radius:9px; border:1px solid var(--line); background:transparent; color:var(--ink); cursor:pointer; display:inline-flex; align-items:center; justify-content:center;}
|
||||
|
||||
.body{padding:12px; display:flex; flex-direction:column; gap:12px;}
|
||||
|
||||
/* CARDURI compacte — doar numere, un rand */
|
||||
.stats{display:flex; background:var(--card2); border:1px solid var(--line); border-radius:11px; overflow:hidden;}
|
||||
.stat{flex:1; text-align:center; padding:10px 4px; border-right:1px solid var(--line2);}
|
||||
.stat:last-child{border-right:none;}
|
||||
.stat .n{font-size:var(--fs-xl); font-weight:700; line-height:1;}
|
||||
.stat .l{font-size:11px; color:var(--muted); margin-top:4px;}
|
||||
.s-ok{color:var(--ok);} .s-acc{color:var(--accent);} .s-err{color:var(--err);}
|
||||
|
||||
/* IMPORT colapsat */
|
||||
.import-collapse{border:1px solid var(--line); border-radius:11px; background:var(--card); overflow:hidden;}
|
||||
.import-collapse>summary{list-style:none; cursor:pointer; display:flex; align-items:center; justify-content:space-between; gap:8px; padding:13px 14px; font-size:var(--fs-base); font-weight:600; color:var(--ink); min-height:48px;}
|
||||
.import-collapse>summary::-webkit-details-marker{display:none;}
|
||||
.import-collapse>summary .ic-l{display:flex; align-items:center; gap:9px;}
|
||||
.import-collapse .plus{display:inline-flex; width:24px; height:24px; align-items:center; justify-content:center; border-radius:7px; background:color-mix(in srgb,var(--accent) 16%,transparent); color:var(--accent); font-size:17px; line-height:1;}
|
||||
.import-collapse>summary .chev{font-size:var(--fs-sm); color:var(--muted);}
|
||||
|
||||
/* NAV */
|
||||
.subnav{display:flex; gap:6px; border-bottom:1px solid var(--line);}
|
||||
.subnav a{flex:1; text-align:center; font-size:var(--fs-sm); font-weight:600; padding:10px 0; border-radius:9px 9px 0 0; color:var(--muted); text-decoration:none; border:1px solid transparent; border-bottom:none; margin-bottom:-1px;}
|
||||
.subnav a.active{color:var(--ink); background:var(--card); border-color:var(--line); border-bottom:1px solid var(--card);}
|
||||
.badge{display:inline-flex; align-items:center; justify-content:center; min-width:18px; height:18px; margin-left:5px; padding:0 5px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;}
|
||||
|
||||
/* LISTA — filtre se ASEAZA pe randuri (wrap), FARA linie de scroll */
|
||||
.panel{background:var(--card); border:1px solid var(--line); border-radius:0 11px 11px 11px; overflow:hidden;}
|
||||
.filtre{display:flex; gap:7px; flex-wrap:wrap; padding:11px 12px; border-bottom:1px solid var(--line2);}
|
||||
.pillf{font-size:var(--fs-sm); padding:7px 14px; border-radius:99px; border:1px solid var(--line); background:transparent; color:var(--muted);}
|
||||
.pillf.on{background:color-mix(in srgb,var(--accent) 16%,transparent); border-color:transparent; color:var(--accent); font-weight:600;}
|
||||
.rand{display:flex; align-items:center; justify-content:space-between; gap:10px; padding:13px 13px; border-bottom:1px solid var(--line2); min-height:56px;}
|
||||
.rand:last-child{border-bottom:none;}
|
||||
.slim-vin{font-family:var(--font-mono); font-size:var(--fs-md); font-weight:500;}
|
||||
.slim-meta{font-size:var(--fs-sm); color:var(--muted); margin-top:3px;}
|
||||
.pill{display:inline-flex; align-items:center; gap:6px; padding:5px 11px; border-radius:99px; font-size:var(--fs-sm); font-weight:500; flex-shrink:0;}
|
||||
.pill .pdot{width:7px; height:7px; border-radius:99px;}
|
||||
.pill.sent{background:color-mix(in srgb,var(--ok) 14%,transparent); color:var(--ok);} .pill.sent .pdot{background:var(--ok);}
|
||||
.pill.coada{background:color-mix(in srgb,var(--accent) 16%,transparent); color:var(--accent);} .pill.coada .pdot{background:var(--accent);}
|
||||
.pill.err{background:color-mix(in srgb,var(--err) 14%,transparent); color:var(--err);} .pill.err .pdot{background:var(--err);}
|
||||
|
||||
/* meniu burger deschis */
|
||||
.scrim{position:absolute; inset:0; background:rgba(0,0,0,.45); z-index:8;}
|
||||
.menu{position:absolute; top:52px; right:10px; width:240px; background:var(--card); border:1px solid var(--line); border-radius:12px; box-shadow:0 20px 50px -16px rgba(0,0,0,.7); padding:7px; z-index:9;}
|
||||
.menu-status{display:flex; align-items:center; gap:9px; padding:11px 11px; font-size:var(--fs-base); font-weight:600; color:var(--ok);}
|
||||
.menu-status .d{width:10px; height:10px; border-radius:99px; background:var(--ok); box-shadow:0 0 0 4px color-mix(in srgb,var(--ok) 22%,transparent);}
|
||||
.menu-status small{font-weight:400; color:var(--muted); font-family:var(--font-mono); font-size:11px;}
|
||||
.menu-plan{display:flex; align-items:center; justify-content:space-between; padding:6px 11px 8px; font-size:var(--fs-sm); color:var(--muted);}
|
||||
.menu-plan b{color:var(--accent);} .menu-plan .trial{font-size:11px;}
|
||||
.menu a{display:flex; align-items:center; justify-content:space-between; padding:12px 11px; border-radius:8px; font-size:var(--fs-base); color:var(--ink); text-decoration:none;}
|
||||
.menu a:active{background:var(--card2);}
|
||||
.menu hr{border:none; border-top:1px solid var(--line); margin:5px 4px;}
|
||||
|
||||
/* ecran editare full-screen */
|
||||
.modal-head{display:flex; align-items:center; justify-content:space-between; height:56px; padding:0 12px; border-bottom:1px solid var(--line); background:var(--hbg); position:sticky; top:0; z-index:5;}
|
||||
.modal-head .t{font-size:var(--fs-md); font-weight:700;}
|
||||
.field{margin-bottom:14px;}
|
||||
.field label{display:block; font-size:var(--fs-sm); color:var(--muted); margin-bottom:6px;}
|
||||
.field input, .field textarea, .field select{width:100%; font-family:var(--font-ui); font-size:var(--fs-md); color:var(--ink); background:var(--card2); border:1px solid var(--line); border-radius:9px; padding:11px 13px; min-height:46px;}
|
||||
.field input.mono{font-family:var(--font-mono);}
|
||||
.grid2{display:grid; grid-template-columns:1fr 1fr; gap:10px;}
|
||||
.op-row{padding:11px 0; border-bottom:1px solid var(--line2);}
|
||||
.op-name{font-size:var(--fs-md); font-weight:600; display:block; margin-bottom:8px;} .op-name small{font-weight:400; color:var(--muted); font-size:var(--fs-sm);}
|
||||
.op-ctl{display:flex; align-items:center; gap:8px;}
|
||||
.chip{display:inline-flex; align-items:center; gap:6px; font-family:var(--font-mono); font-size:var(--fs-sm); background:color-mix(in srgb,var(--accent) 18%,transparent); color:var(--accent); padding:7px 11px; border-radius:8px;}
|
||||
.chip button{background:none; border:none; color:inherit; cursor:pointer; font-size:var(--fs-md);}
|
||||
.addcode{width:100%; font-size:var(--fs-sm); border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line)); background:transparent; color:var(--accent); border-radius:9px; padding:11px; cursor:pointer; margin-top:10px;}
|
||||
.actrow{display:flex; flex-direction:column; gap:10px; margin-top:18px;}
|
||||
.btn-primary{width:100%; font-size:var(--fs-md); font-weight:600; height:46px; background:var(--accent); color:#fff; border:none; border-radius:9px; cursor:pointer;}
|
||||
.btn-ghost{width:100%; font-size:var(--fs-md); height:46px; background:transparent; color:var(--ink); border:1px solid var(--line); border-radius:9px; cursor:pointer;}
|
||||
</style>
|
||||
</head>
|
||||
<body data-theme="grafit">
|
||||
<div class="stage">
|
||||
|
||||
<!-- ECRAN 1: DASHBOARD curat (RAR dot in antet, fara linie de scroll la filtre) -->
|
||||
<div>
|
||||
<div class="phone"><div class="screen"><div class="scroll">
|
||||
<header>
|
||||
<span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span>
|
||||
<div class="h-center">
|
||||
<div class="h-title">ROMFAST <span class="accent">AUTOPASS</span><span class="tier">Pro</span></div>
|
||||
<div class="h-sub">Service auto: <span class="svc">Service Auto Vâlcea SRL</span></div>
|
||||
</div>
|
||||
<div class="h-right">
|
||||
<span class="rar-dot" title="RAR online · ultima autentificare 28.06.2026 09:41"><span class="d"></span></span>
|
||||
<button class="icon-btn" title="Temă: Grafit"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg></button>
|
||||
<button class="icon-btn" title="Meniu">☰</button>
|
||||
</div>
|
||||
</header>
|
||||
<div class="body">
|
||||
<div class="stats">
|
||||
<div class="stat"><div class="n s-ok">847</div><div class="l">Total</div></div>
|
||||
<div class="stat"><div class="n s-ok">124</div><div class="l">Lună</div></div>
|
||||
<div class="stat"><div class="n s-ok">9</div><div class="l">Azi</div></div>
|
||||
<div class="stat"><div class="n s-acc">12</div><div class="l">Coadă</div></div>
|
||||
<div class="stat"><div class="n s-err">2</div><div class="l">Corectat</div></div>
|
||||
</div>
|
||||
<details class="import-collapse">
|
||||
<summary><span class="ic-l"><span class="plus">+</span> Importă fișier (XLSX / CSV)</span><span class="chev">▾</span></summary>
|
||||
</details>
|
||||
<div>
|
||||
<div class="subnav">
|
||||
<a href="#" class="active">Trimiteri</a>
|
||||
<a href="#">Mapări <span class="badge">2</span></a>
|
||||
</div>
|
||||
<div class="panel">
|
||||
<div class="filtre">
|
||||
<button class="pillf on">Toate</button>
|
||||
<button class="pillf">În coadă</button>
|
||||
<button class="pillf">Trimise</button>
|
||||
<button class="pillf">De corectat</button>
|
||||
</div>
|
||||
<div class="rand"><div><div class="slim-vin">WBA8E9...K7F2</div><div class="slim-meta">Inspecție tehnică · 09:42</div></div><span class="pill sent"><span class="pdot"></span>Trimis</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">WVWZZZ...3M1</div><div class="slim-meta">Revizie periodică · 09:38</div></div><span class="pill coada"><span class="pdot"></span>În coadă</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">VF1RFB...A88</div><div class="slim-meta">Sistem frânare · 09:31</div></div><span class="pill err"><span class="pdot"></span>De corectat</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">ZAR937...C04</div><div class="slim-meta">Schimb ulei · 09:24</div></div><span class="pill sent"><span class="pdot"></span>Trimis</span></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div></div></div>
|
||||
<div class="cap">390px · Acasă — RAR online = dot în antet (dată/oră pe hover), filtre fără linie de scroll</div>
|
||||
</div>
|
||||
|
||||
<!-- ECRAN 2: meniu burger deschis (RAR online si aici) -->
|
||||
<div>
|
||||
<div class="phone"><div class="screen">
|
||||
<header>
|
||||
<span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span>
|
||||
<div class="h-center">
|
||||
<div class="h-title">ROMFAST <span class="accent">AUTOPASS</span><span class="tier">Pro</span></div>
|
||||
<div class="h-sub">Service auto: <span class="svc">Service Auto Vâlcea SRL</span></div>
|
||||
</div>
|
||||
<div class="h-right">
|
||||
<span class="rar-dot" title="RAR online"><span class="d"></span></span>
|
||||
<button class="icon-btn" title="Temă: Grafit"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg></button>
|
||||
<button class="icon-btn" title="Închide meniu">×</button>
|
||||
</div>
|
||||
</header>
|
||||
<div class="scrim"></div>
|
||||
<div class="menu">
|
||||
<div class="menu-status"><span class="d"></span> RAR online <small>· 09:41</small></div>
|
||||
<div class="menu-plan">Plan: <b>Pro</b> <span class="trial">trial · 18 zile</span></div>
|
||||
<hr>
|
||||
<a href="#">Trimiteri</a>
|
||||
<a href="#">Mapări <span class="badge">2</span></a>
|
||||
<hr>
|
||||
<a href="#">Nomenclator</a>
|
||||
<hr>
|
||||
<a href="#">Cont</a>
|
||||
<a href="#">Integrare</a>
|
||||
<a href="#">Jurnal</a>
|
||||
<hr>
|
||||
<a href="#">Ieși din cont</a>
|
||||
</div>
|
||||
</div></div>
|
||||
<div class="cap">390px · Meniu burger — RAR online + Plan (Pro) + separatoare între secțiuni</div>
|
||||
</div>
|
||||
|
||||
<!-- ECRAN 3: editare full-screen (trimitere nefinalizata) -->
|
||||
<div>
|
||||
<div class="phone"><div class="screen"><div class="scroll">
|
||||
<div class="modal-head"><span class="t">Corectează trimiterea</span><button class="icon-btn" title="Închide">×</button></div>
|
||||
<div class="body" style="gap:0;">
|
||||
<div class="field"><label>VIN (serie șasiu)</label><input class="mono" value="VF1RFB00A88142073"></div>
|
||||
<div class="grid2">
|
||||
<div class="field"><label>Data prestației</label><input class="mono" value="2026-06-22"></div>
|
||||
<div class="field"><label>Nr. înmatriculare</label><input class="mono" value="CT88NOE"></div>
|
||||
</div>
|
||||
<div class="field"><label>Observații (operațiile efectuate)</label><textarea rows="2">Schimbare plăcuțe frână față</textarea></div>
|
||||
<div class="field" style="margin-bottom:6px;">
|
||||
<label>Prestații — cod RAR pe fiecare operație</label>
|
||||
<div class="op-row"><span class="op-name">REVIZIE PERIODICĂ <small>— la 15.000 km</small></span><div class="op-ctl"><span class="chip">REV2 <button>×</button></span></div></div>
|
||||
<div class="op-row" style="border-left:2px solid var(--warn); padding-left:10px;"><span class="op-name">SCHIMB PLĂCUȚE FRÂNĂ <small style="color:var(--warn)">— lipsă cod</small></span><div class="op-ctl"><select><option>— alege cod RAR —</option><option>FRN1 — Sistem de frânare</option></select></div></div>
|
||||
<button class="addcode">+ Adaugă altă operație / cod RAR</button>
|
||||
</div>
|
||||
<div class="actrow"><button class="btn-primary">Salvează și retrimite</button><button class="btn-ghost">Renunță</button></div>
|
||||
</div>
|
||||
</div></div></div>
|
||||
<div class="cap">390px · Editare full-screen — trimitere nefinalizată (picker cod+denumire, Renunță)</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
241
docs/mockups/prd-5.16-dashboard.html
Normal file
241
docs/mockups/prd-5.16-dashboard.html
Normal file
@@ -0,0 +1,241 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>PRD 5.16 — Dashboard aplicatie (compact, minimalist)</title>
|
||||
<style>
|
||||
:root{
|
||||
--font-ui: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
||||
--font-mono: ui-monospace, "SF Mono", "Cascadia Code", "Segoe UI Mono", "Roboto Mono", Menlo, Consolas, monospace;
|
||||
--fs-xs:12px; --fs-sm:13.5px; --fs-base:15px; --fs-md:16px; --fs-lg:18px; --fs-xl:20px; --fs-2xl:28px;
|
||||
--bg:#0f1218; --card:#181c24; --card2:#0f1218; --ink:#e6e9ef; --muted:#8b93a7;
|
||||
--line:#262b36; --line2:#1f2530; --accent:#6ea2ec; --ok:#2FBF8F; --warn:#E0A93B; --err:#E05D5D;
|
||||
--hbg:rgba(15,18,24,.9);
|
||||
}
|
||||
body[data-theme="hartie"]{ --bg:#f3efe6; --card:#fffdf7; --card2:#f3efe6; --ink:#1e1a13; --muted:#6a6052; --line:#e2dccc; --line2:#ece6d9; --accent:#1F5FBF; --ok:#1c7d5d; --warn:#b45309; --err:#bd463c; --hbg:rgba(255,253,247,.92); }
|
||||
body[data-theme="cobalt"]{ --bg:#080d1c; --card:#111a33; --card2:#0b1226; --ink:#e9ecfb; --muted:#8a93b8; --line:#1d2747; --line2:#161f3a; --accent:#8aa0ff; --ok:#2fd0a6; --err:#f06a7a; --hbg:rgba(8,13,28,.92); }
|
||||
body[data-theme="cupru"]{ --bg:#15110b; --card:#211a12; --card2:#15110b; --ink:#efe6d6; --muted:#a89a85; --line:#36291c; --line2:#281e14; --accent:#dfa45c; --ok:#67b98c; --err:#e2685a; --hbg:rgba(21,17,11,.92); }
|
||||
*{box-sizing:border-box;}
|
||||
body{margin:0; background:var(--bg); color:var(--ink); font-family:var(--font-ui); font-size:var(--fs-base); line-height:1.55; -webkit-font-smoothing:antialiased;}
|
||||
.mono{font-family:var(--font-mono);} .muted{color:var(--muted);}
|
||||
|
||||
/* HEADER branded (numele service e DOAR aici, nu se mai duplica jos) */
|
||||
header{position:sticky; top:0; z-index:5; display:grid; grid-template-columns:1fr auto 1fr; align-items:center; gap:16px; height:64px; padding:0 22px; background:var(--hbg); backdrop-filter:blur(8px); border-bottom:1px solid var(--line);}
|
||||
.logo-fallback{display:inline-flex; align-items:center; gap:6px; font-weight:800; font-size:var(--fs-lg);}
|
||||
.logo-fallback .rom{color:#D1342F;} .logo-fallback .fast{color:var(--accent);}
|
||||
.h-center{text-align:center; line-height:1.15;}
|
||||
.h-title{font-size:var(--fs-md); font-weight:700;} .h-title .accent{color:var(--accent);}
|
||||
.h-sub{font-size:var(--fs-xs); color:var(--muted); margin-top:2px;} .h-sub .svc{color:var(--ink); font-weight:600;}
|
||||
.env{display:inline-block; margin-left:8px; padding:1px 7px; border-radius:99px; font-size:10px; font-weight:700; text-transform:uppercase; letter-spacing:.04em; color:var(--warn); background:color-mix(in srgb,var(--warn) 16%,transparent);}
|
||||
/* badge tip cont (Gratuit/Standard/Pro/Premium) */
|
||||
.tier{display:inline-block; margin-left:6px; padding:1px 8px; border-radius:99px; font-size:10px; font-weight:700; text-transform:uppercase; letter-spacing:.04em; color:var(--accent); background:color-mix(in srgb,var(--accent) 16%,transparent);}
|
||||
.h-right{display:flex; align-items:center; justify-content:flex-end; gap:10px;}
|
||||
/* dot RAR online compact in antet (inlocuieste banda) — datetime pe title/hover */
|
||||
.rar-chip{display:flex; align-items:center; gap:8px; height:38px; padding:0 13px; border-radius:99px; border:1px solid color-mix(in srgb,var(--ok) 35%,var(--line)); background:color-mix(in srgb,var(--ok) 10%,transparent); color:var(--ok); font-size:var(--fs-sm); font-weight:600; cursor:default;}
|
||||
.rar-chip.blocat{border-color:color-mix(in srgb,var(--err) 45%,var(--line)); background:color-mix(in srgb,var(--err) 12%,transparent); color:var(--err);}
|
||||
.rar-chip .dot{width:9px; height:9px; border-radius:99px; background:currentColor; box-shadow:0 0 0 4px color-mix(in srgb,currentColor 22%,transparent);}
|
||||
.tema-btn{display:flex; align-items:center; gap:8px; height:38px; padding:0 13px; border-radius:8px; background:transparent; border:1px solid var(--line); color:var(--muted); font-family:var(--font-ui); font-size:var(--fs-sm); cursor:pointer;}
|
||||
.tema-btn:hover{border-color:var(--accent); color:var(--ink);}
|
||||
.ver{font-size:var(--fs-xs); color:var(--muted);}
|
||||
.icon-btn{width:38px; height:38px; border-radius:8px; border:1px solid var(--line); background:transparent; color:var(--ink); font-size:18px; cursor:pointer; display:inline-flex; align-items:center; justify-content:center; position:relative;}
|
||||
/* meniu burger deschis (mockup) — contine si starea RAR */
|
||||
.menu{position:absolute; top:46px; right:0; width:230px; background:var(--card); border:1px solid var(--line); border-radius:10px; box-shadow:0 18px 40px -16px rgba(0,0,0,.6); padding:6px; z-index:10; text-align:left;}
|
||||
.menu-status{display:flex; align-items:center; gap:8px; padding:9px 10px; font-size:var(--fs-sm); font-weight:600; color:var(--ok);}
|
||||
.menu-status small{font-weight:400; color:var(--muted); font-family:var(--font-mono); font-size:11px;}
|
||||
.menu-plan{display:flex; align-items:center; justify-content:space-between; padding:8px 10px 4px; font-size:var(--fs-sm); color:var(--muted);}
|
||||
.menu-plan b{color:var(--accent);}
|
||||
.menu-plan .trial{font-size:11px; color:var(--muted);}
|
||||
.menu a{display:flex; align-items:center; justify-content:space-between; padding:9px 10px; border-radius:7px; font-size:var(--fs-sm); color:var(--ink); text-decoration:none;}
|
||||
.menu a:hover{background:var(--card2);}
|
||||
.menu hr{border:none; border-top:1px solid var(--line); margin:5px 4px;}
|
||||
.menu .badge{display:inline-flex; align-items:center; justify-content:center; min-width:18px; height:18px; padding:0 5px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;}
|
||||
|
||||
.wrap{max-width:1000px; margin:0 auto; padding:16px 22px 70px; display:flex; flex-direction:column; gap:14px;}
|
||||
|
||||
/* Banda de stare — APARE DOAR cand e blocat (zero-silent-failures) */
|
||||
.strip{display:flex; align-items:center; justify-content:space-between; gap:12px; padding:12px 16px; border-radius:10px;
|
||||
background:color-mix(in srgb, var(--ok) 13%, transparent); border:1px solid color-mix(in srgb, var(--ok) 30%, transparent);}
|
||||
.strip.blocat{background:color-mix(in srgb, var(--err) 13%, transparent); border-color:color-mix(in srgb, var(--err) 35%, transparent); color:var(--err);}
|
||||
.strip-left{display:flex; align-items:center; gap:10px; font-weight:700; font-size:var(--fs-md);}
|
||||
.strip .dot{width:10px; height:10px; border-radius:99px; background:var(--ok); flex-shrink:0; box-shadow:0 0 0 4px color-mix(in srgb, var(--ok) 22%, transparent);}
|
||||
.strip.blocat .dot{background:var(--err); box-shadow:0 0 0 4px color-mix(in srgb, var(--err) 22%, transparent);}
|
||||
.strip-right{font-family:var(--font-mono); font-size:var(--fs-xs); color:var(--muted);}
|
||||
|
||||
/* 2. CARDURI contor — standalone, fara titlu de sectiune */
|
||||
.contoare{display:grid; grid-template-columns:repeat(5,1fr); gap:10px;}
|
||||
.contor-card{background:var(--card2); border:1px solid var(--line); border-radius:10px; padding:14px 16px;}
|
||||
.contor-card.primar{border-color:color-mix(in srgb,var(--ok) 40%,var(--line));}
|
||||
.contor-cifra{font-size:var(--fs-2xl); font-weight:700; line-height:1;}
|
||||
.contor-label{font-size:var(--fs-sm); color:var(--muted); margin-top:7px;}
|
||||
.s-ok{color:var(--ok);} .s-acc{color:var(--accent);} .s-err{color:var(--err);}
|
||||
|
||||
/* 3. IMPORT colapsat */
|
||||
.import-collapse{border:1px solid var(--line); border-radius:10px; background:var(--card); overflow:hidden;}
|
||||
.import-collapse>summary{list-style:none; cursor:pointer; display:flex; align-items:center; justify-content:space-between; gap:10px; padding:12px 16px; font-size:var(--fs-sm); font-weight:600; color:var(--ink);}
|
||||
.import-collapse>summary::-webkit-details-marker{display:none;}
|
||||
.import-collapse>summary .ic-l{display:flex; align-items:center; gap:10px;}
|
||||
.import-collapse .plus{display:inline-flex; width:24px; height:24px; align-items:center; justify-content:center; border-radius:7px; background:color-mix(in srgb,var(--accent) 16%,transparent); color:var(--accent); font-size:17px; line-height:1;}
|
||||
.import-collapse>summary .ic-r{font-size:var(--fs-xs); color:var(--muted);}
|
||||
.import-collapse[open]>summary{border-bottom:1px solid var(--line);}
|
||||
.import-body{display:flex; align-items:center; justify-content:space-between; gap:14px; padding:16px; border:1px dashed color-mix(in srgb,var(--accent) 45%,var(--line)); border-radius:10px; margin:12px;}
|
||||
.import-body .u-tx{font-size:var(--fs-md); font-weight:600;}
|
||||
.import-body .u-sub{font-size:var(--fs-sm); color:var(--muted); margin-top:2px;}
|
||||
.btn-primary{font-family:var(--font-ui); font-size:var(--fs-md); font-weight:600; height:42px; padding:0 20px; background:var(--accent); color:#fff; border:none; border-radius:8px; cursor:pointer;}
|
||||
|
||||
/* 4. NAV tab-uri Trimiteri / Mapari */
|
||||
.subnav{display:flex; gap:6px; border-bottom:1px solid var(--line);}
|
||||
.subnav a{font-size:var(--fs-sm); font-weight:600; padding:9px 16px; border-radius:8px 8px 0 0; color:var(--muted); text-decoration:none; border:1px solid transparent; border-bottom:none; margin-bottom:-1px;}
|
||||
.subnav a.active{color:var(--ink); background:var(--card); border-color:var(--line); border-bottom:1px solid var(--card);}
|
||||
.subnav .badge{display:inline-flex; align-items:center; justify-content:center; min-width:18px; height:18px; margin-left:6px; padding:0 5px; border-radius:99px; background:var(--err); color:#fff; font-size:11px; font-weight:700;}
|
||||
|
||||
/* 5. LISTA (fara titlu/subtitlu de sectiune) */
|
||||
.panel{background:var(--card); border:1px solid var(--line); border-radius:0 12px 12px 12px; overflow:hidden;}
|
||||
.filtre{display:flex; gap:8px; padding:12px 16px; flex-wrap:wrap; border-bottom:1px solid var(--line2);}
|
||||
.pillf{font-size:var(--fs-sm); padding:6px 13px; border-radius:99px; border:1px solid var(--line); background:transparent; color:var(--muted); cursor:pointer;}
|
||||
.pillf.on{background:color-mix(in srgb,var(--accent) 16%,transparent); border-color:transparent; color:var(--accent); font-weight:600;}
|
||||
.rand{display:flex; align-items:center; justify-content:space-between; padding:13px 16px; border-bottom:1px solid var(--line2); cursor:pointer;}
|
||||
.rand:hover{background:color-mix(in srgb,var(--accent) 6%,transparent);}
|
||||
.rand:last-child{border-bottom:none;}
|
||||
.slim-vin{font-family:var(--font-mono); font-size:var(--fs-md); font-weight:500;}
|
||||
.slim-meta{font-size:var(--fs-sm); color:var(--muted); margin-top:3px;}
|
||||
.pill{display:inline-flex; align-items:center; gap:7px; padding:5px 12px; border-radius:99px; font-size:var(--fs-sm); font-weight:500;}
|
||||
.pill .pdot{width:7px; height:7px; border-radius:99px;}
|
||||
.pill.sent{background:color-mix(in srgb,var(--ok) 14%,transparent); color:var(--ok);} .pill.sent .pdot{background:var(--ok);}
|
||||
.pill.coada{background:color-mix(in srgb,var(--accent) 16%,transparent); color:var(--accent);} .pill.coada .pdot{background:var(--accent);}
|
||||
.pill.err{background:color-mix(in srgb,var(--err) 14%,transparent); color:var(--err);} .pill.err .pdot{background:var(--err);}
|
||||
|
||||
/* MODAL editare trimitere nefinalizata (la click pe rand) */
|
||||
.editmodal{max-width:560px; background:var(--card); border:1px solid var(--line); border-radius:12px; overflow:hidden;}
|
||||
.editmodal .mhead{display:flex; align-items:center; justify-content:space-between; padding:14px 18px; border-bottom:1px solid var(--line);}
|
||||
.editmodal .mhead .t{font-size:var(--fs-md); font-weight:700;}
|
||||
.editmodal .mbody{padding:18px;}
|
||||
.field{margin-bottom:14px;}
|
||||
.field label{display:block; font-size:var(--fs-sm); color:var(--muted); margin-bottom:6px;}
|
||||
.field input, .field textarea, .field select{width:100%; font-family:var(--font-ui); font-size:var(--fs-md); color:var(--ink); background:var(--card2); border:1px solid var(--line); border-radius:8px; padding:9px 12px; min-height:40px;}
|
||||
.field input.mono{font-family:var(--font-mono);}
|
||||
.grid2{display:grid; grid-template-columns:1fr 1fr; gap:12px;}
|
||||
.op-row{display:flex; align-items:center; justify-content:space-between; gap:10px; padding:10px 0; border-bottom:1px solid var(--line2);}
|
||||
.op-name{font-size:var(--fs-md); font-weight:600;} .op-name small{font-weight:400; color:var(--muted); font-size:var(--fs-sm);}
|
||||
.chip{display:inline-flex; align-items:center; gap:6px; font-family:var(--font-mono); font-size:var(--fs-sm); background:color-mix(in srgb,var(--accent) 18%,transparent); color:var(--accent); padding:5px 10px; border-radius:7px;}
|
||||
.chip button{background:none; border:none; color:inherit; cursor:pointer; font-size:var(--fs-md);}
|
||||
.addcode{font-size:var(--fs-sm); border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line)); background:transparent; color:var(--accent); border-radius:7px; padding:6px 12px; cursor:pointer;}
|
||||
.btn-ghost{font-size:var(--fs-md); height:42px; padding:0 18px; background:transparent; color:var(--ink); border:1px solid var(--line); border-radius:8px; cursor:pointer;}
|
||||
.actrow{display:flex; gap:10px; margin-top:16px;}
|
||||
</style>
|
||||
</head>
|
||||
<body data-theme="grafit">
|
||||
|
||||
<header>
|
||||
<div><span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span></div>
|
||||
<div class="h-center">
|
||||
<div class="h-title">ROMFAST <span class="accent">AUTOPASS</span><span class="env">test</span><span class="tier">Pro</span></div>
|
||||
<div class="h-sub">Service auto: <span class="svc">Service Auto Vâlcea SRL</span></div>
|
||||
</div>
|
||||
<div class="h-right">
|
||||
<div class="rar-chip" title="Ultima autentificare RAR: 28.06.2026 09:41"><span class="dot"></span> RAR online</div>
|
||||
<button class="tema-btn" onclick="cycle()">
|
||||
<svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg>
|
||||
<span id="t-label">Grafit</span>
|
||||
</button>
|
||||
<span class="ver">v5.16</span>
|
||||
<button class="icon-btn" title="Meniu cont">☰
|
||||
<div class="menu">
|
||||
<div class="menu-status"><span class="rar-chip" style="height:auto;padding:0;border:none;background:none;"><span class="dot"></span></span> RAR online <small>· 09:41</small></div>
|
||||
<div class="menu-plan">Plan: <b>Pro</b> <span class="trial">trial · 18 zile rămase</span></div>
|
||||
<hr>
|
||||
<a href="#">Trimiteri</a>
|
||||
<a href="#">Mapări <span class="badge">2</span></a>
|
||||
<hr>
|
||||
<a href="#">Nomenclator</a>
|
||||
<hr>
|
||||
<a href="#">Cont</a>
|
||||
<a href="#">Integrare</a>
|
||||
<a href="#">Jurnal</a>
|
||||
<hr>
|
||||
<a href="#">Ieși din cont</a>
|
||||
</div>
|
||||
</button>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
<div class="wrap">
|
||||
|
||||
<!-- CARDURI (fara titlu de sectiune; RAR online e acum dot in antet) -->
|
||||
<div class="contoare">
|
||||
<div class="contor-card primar"><div class="contor-cifra s-ok">847</div><div class="contor-label">Total trimise</div></div>
|
||||
<div class="contor-card"><div class="contor-cifra s-ok">124</div><div class="contor-label">Luna asta</div></div>
|
||||
<div class="contor-card"><div class="contor-cifra s-ok">9</div><div class="contor-label">Azi</div></div>
|
||||
<div class="contor-card"><div class="contor-cifra s-acc">12</div><div class="contor-label">În coadă</div></div>
|
||||
<div class="contor-card"><div class="contor-cifra s-err">2</div><div class="contor-label">De corectat</div></div>
|
||||
</div>
|
||||
|
||||
<!-- 3. IMPORT colapsat -->
|
||||
<details class="import-collapse">
|
||||
<summary>
|
||||
<span class="ic-l"><span class="plus">+</span> Importă fișier (XLSX / CSV)</span>
|
||||
<span class="ic-r">trage-l aici sau apasă pentru a deschide ▾</span>
|
||||
</summary>
|
||||
<div class="import-body">
|
||||
<div><div class="u-tx">Încarcă un fișier sau trage-l aici</div><div class="u-sub">Mapezi coloanele o singură dată — apoi trimitem la RAR automat.</div></div>
|
||||
<button class="btn-primary">Alege fișier</button>
|
||||
</div>
|
||||
</details>
|
||||
|
||||
<!-- 4 + 5. NAV + LISTA -->
|
||||
<div>
|
||||
<div class="subnav">
|
||||
<a href="#" class="active">Trimiteri</a>
|
||||
<a href="#">Mapări <span class="badge">2</span></a>
|
||||
</div>
|
||||
<div class="panel">
|
||||
<div class="filtre">
|
||||
<button class="pillf on">Toate</button>
|
||||
<button class="pillf">În coadă</button>
|
||||
<button class="pillf">Trimise</button>
|
||||
<button class="pillf">De corectat</button>
|
||||
</div>
|
||||
<div class="rand"><div><div class="slim-vin">WBA8E9...K7F2</div><div class="slim-meta">Inspecție tehnică · 09:42</div></div><span class="pill sent"><span class="pdot"></span>Trimis</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">WVWZZZ...3M1</div><div class="slim-meta">Revizie periodică · 09:38</div></div><span class="pill coada"><span class="pdot"></span>În coadă</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">VF1RFB...A88</div><div class="slim-meta">Sistem frânare · 09:31</div></div><span class="pill err"><span class="pdot"></span>De corectat</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">ZAR937...C04</div><div class="slim-meta">Schimb ulei · 09:24</div></div><span class="pill sent"><span class="pdot"></span>Trimis</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">JTDBR...9920</div><div class="slim-meta">Inspecție tehnică · 09:18</div></div><span class="pill sent"><span class="pdot"></span>Trimis</span></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- DOAR cand e BLOCAT: banda rosie reapare (zero-silent-failures) -->
|
||||
<div style="margin-top:18px; font-size:var(--fs-xs); text-transform:uppercase; letter-spacing:.08em; color:var(--err); font-weight:700;">Stare BLOCAT — banda apare DOAR atunci (worker oprit / RAR inaccesibil)</div>
|
||||
<div class="strip blocat">
|
||||
<span class="strip-left"><span class="dot"></span> Blocat: RAR inaccesibil — declarațiile NU pleacă</span>
|
||||
<span class="strip-right">Ultima autentificare RAR: 28.06.2026 09:41</span>
|
||||
</div>
|
||||
|
||||
<!-- MODAL editare: apare la click pe o trimitere nefinalizata (needs_data / needs_mapping / error) -->
|
||||
<div style="margin-top:22px; font-size:var(--fs-xs); text-transform:uppercase; letter-spacing:.08em; color:var(--accent); font-weight:700;">Modal editare — la click pe o trimitere nefinalizată (needs_data / needs_mapping)</div>
|
||||
<div class="editmodal" style="margin-top:8px;">
|
||||
<div class="mhead"><span class="t">Corectează trimiterea</span><button class="icon-btn" title="Închide">×</button></div>
|
||||
<div class="mbody">
|
||||
<div class="field"><label>VIN (serie șasiu)</label><input class="mono" value="VF1RFB00A88142073"></div>
|
||||
<div class="grid2">
|
||||
<div class="field"><label>Data prestației</label><input class="mono" value="2026-06-22"></div>
|
||||
<div class="field"><label>Număr înmatriculare</label><input class="mono" value="CT88NOE"></div>
|
||||
</div>
|
||||
<div class="field"><label>Observații (operațiile efectuate)</label><textarea rows="2">Schimbare plăcuțe frână față</textarea></div>
|
||||
<div class="field">
|
||||
<label>Prestații — cod RAR pe fiecare operație</label>
|
||||
<div class="op-row"><span class="op-name">REVIZIE PERIODICĂ <small>— la 15.000 km</small></span><span class="chip">REV2 <button>×</button></span></div>
|
||||
<div class="op-row" style="border-left:2px solid var(--warn); padding-left:10px;"><span class="op-name">SCHIMB PLĂCUȚE FRÂNĂ <small style="color:var(--warn)">— lipsă cod</small></span><select><option>— alege cod RAR —</option><option>FRN1 — Sistem de frânare</option><option>REV2 — Revizie periodică</option></select></div>
|
||||
<div style="margin-top:10px;"><button class="addcode">+ Adaugă altă operație / cod RAR</button></div>
|
||||
</div>
|
||||
<div class="actrow"><button class="btn-primary">Salvează și retrimite</button><button class="btn-ghost">Renunță</button></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
var THEMES=[['grafit','Grafit'],['cobalt','Cobalt'],['cupru','Cupru'],['hartie','Hârtie']];
|
||||
var i=0;
|
||||
function cycle(){ i=(i+1)%THEMES.length; document.body.setAttribute('data-theme',THEMES[i][0]); document.getElementById('t-label').textContent=THEMES[i][1]; }
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
187
docs/mockups/prd-5.16-fonturi-system-stack.html
Normal file
187
docs/mockups/prd-5.16-fonturi-system-stack.html
Normal file
@@ -0,0 +1,187 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>PRD 5.16 — Preview fonturi system-stack + scala tipografica</title>
|
||||
<style>
|
||||
/* ============================================================
|
||||
PROPUNERE 5.16: fonturi STANDARD WEB (system font stack).
|
||||
ZERO fisiere de font descarcate. Arata nativ pe fiecare OS.
|
||||
Inlocuieste IBM Plex self-hostat din /static/fonts.
|
||||
============================================================ */
|
||||
:root{
|
||||
/* Stive de font standard web (fara @font-face, fara /static/fonts) */
|
||||
--font-ui: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
||||
--font-mono: ui-monospace, "SF Mono", "Cascadia Code", "Segoe UI Mono", "Roboto Mono", Menlo, Consolas, monospace;
|
||||
|
||||
/* SCALA TIPOGRAFICA UNIFORMA (sursa unica de adevar; azi e ad-hoc 10/11/13px) */
|
||||
--fs-xs: 12px; /* meta, sub-linii mono, hint-uri (azi: 10px) */
|
||||
--fs-sm: 13.5px; /* label-uri formular, pill-uri (azi: 11px) */
|
||||
--fs-base: 15px; /* text body implicit (azi: ~13px) */
|
||||
--fs-md: 16px; /* input-uri, text card (azi: 13px) */
|
||||
--fs-lg: 18px; /* titluri de sectiune mici */
|
||||
--fs-xl: 20px; /* sub-titluri */
|
||||
--fs-2xl: 28px; /* cifra contor (azi: 22px) */
|
||||
--fs-3xl: 34px; /* titlu pagina */
|
||||
--lh-tight: 1.25;
|
||||
--lh-body: 1.55;
|
||||
|
||||
/* paleta grafit (din DESIGN.md) — doar pentru context vizual */
|
||||
--bg:#0f1218; --card:#181c24; --card2:#0f1218; --ink:#e6e9ef; --muted:#8b93a7;
|
||||
--line:#262b36; --line2:#1f2530; --accent:#6ea2ec; --ok:#2FBF8F; --warn:#E0A93B; --err:#E05D5D;
|
||||
}
|
||||
body[data-theme="hartie"]{
|
||||
--bg:#f3efe6; --card:#fffdf7; --card2:#f3efe6; --ink:#1e1a13; --muted:#6a6052;
|
||||
--line:#e2dccc; --line2:#ece6d9; --accent:#1F5FBF; --ok:#1c7d5d; --warn:#b45309; --err:#bd463c;
|
||||
}
|
||||
*{box-sizing:border-box;}
|
||||
body{
|
||||
margin:0; background:var(--bg); color:var(--ink);
|
||||
font-family:var(--font-ui);
|
||||
font-size:var(--fs-base); line-height:var(--lh-body);
|
||||
-webkit-font-smoothing:antialiased;
|
||||
}
|
||||
.wrap{max-width:1100px; margin:0 auto; padding:28px 22px 80px;}
|
||||
.mono{font-family:var(--font-mono);}
|
||||
h1{font-size:var(--fs-3xl); line-height:var(--lh-tight); margin:0 0 6px; letter-spacing:-.02em;}
|
||||
.lead{color:var(--muted); font-size:var(--fs-md); margin:0 0 22px;}
|
||||
.sec{font-size:var(--fs-lg); margin:34px 0 12px; padding-bottom:6px; border-bottom:1px solid var(--line);}
|
||||
.toolbar{display:flex; gap:10px; align-items:center; margin-bottom:8px;}
|
||||
.toolbar button{font-family:var(--font-ui); font-size:var(--fs-sm); height:36px; padding:0 14px;
|
||||
border-radius:7px; border:1px solid var(--line); background:var(--card); color:var(--ink); cursor:pointer;}
|
||||
.note{font-size:var(--fs-sm); color:var(--muted); margin:2px 0 0;}
|
||||
|
||||
/* ---- carduri-contor (aerisite, text mai mare) ---- */
|
||||
.contoare{display:grid; grid-template-columns:repeat(3,1fr); gap:14px;}
|
||||
.contor-card{background:var(--card2); border:1px solid var(--line); border-radius:12px; padding:18px 18px;}
|
||||
.contor-cifra{font-size:var(--fs-2xl); font-weight:700; line-height:1;}
|
||||
.contor-label{font-size:var(--fs-sm); color:var(--muted); margin-top:8px;}
|
||||
.contor-sub{font-family:var(--font-mono); font-size:var(--fs-xs); color:var(--muted); margin-top:4px;}
|
||||
.s-ok{color:var(--ok);} .s-acc{color:var(--accent);} .s-err{color:var(--err);} .s-muted{color:var(--muted);}
|
||||
|
||||
/* ---- strip sanatate cu DOT (nu bifa) pentru RAR online ---- */
|
||||
.strip{display:flex; align-items:center; justify-content:space-between; gap:12px;
|
||||
padding:12px 16px; border-radius:10px; margin-bottom:14px;
|
||||
background:color-mix(in srgb, var(--ok) 13%, transparent);
|
||||
border:1px solid color-mix(in srgb, var(--ok) 30%, transparent);}
|
||||
.strip-left{display:flex; align-items:center; gap:10px; font-weight:700; font-size:var(--fs-md);}
|
||||
.dot{width:10px; height:10px; border-radius:99px; background:var(--ok); flex-shrink:0;
|
||||
box-shadow:0 0 0 4px color-mix(in srgb, var(--ok) 22%, transparent);}
|
||||
.dot.live{animation:pulse 2s ease-in-out infinite;}
|
||||
@keyframes pulse{0%,100%{opacity:1;} 50%{opacity:.55;}}
|
||||
.strip-right{font-family:var(--font-mono); font-size:var(--fs-xs); color:var(--muted);}
|
||||
|
||||
/* ---- lista slim ---- */
|
||||
.lista{background:var(--card); border:1px solid var(--line); border-radius:12px; overflow:hidden; margin-top:14px;}
|
||||
.rand{display:flex; align-items:center; justify-content:space-between; padding:14px 18px; border-bottom:1px solid var(--line2);}
|
||||
.rand:last-child{border-bottom:none;}
|
||||
.slim-vin{font-family:var(--font-mono); font-size:var(--fs-md); font-weight:500;}
|
||||
.slim-meta{font-size:var(--fs-sm); color:var(--muted); margin-top:3px;}
|
||||
.pill{display:inline-flex; align-items:center; gap:7px; padding:5px 12px; border-radius:99px; font-size:var(--fs-sm); font-weight:500;}
|
||||
.pill .pdot{width:7px; height:7px; border-radius:99px;}
|
||||
.pill.sent{background:color-mix(in srgb,var(--ok) 14%,transparent); color:var(--ok);}
|
||||
.pill.sent .pdot{background:var(--ok);}
|
||||
.pill.coada{background:color-mix(in srgb,var(--accent) 16%,transparent); color:var(--accent);}
|
||||
.pill.coada .pdot{background:var(--accent);}
|
||||
.pill.err{background:color-mix(in srgb,var(--err) 14%,transparent); color:var(--err);}
|
||||
.pill.err .pdot{background:var(--err);}
|
||||
|
||||
/* ---- formular editare slim ---- */
|
||||
.form-card{background:var(--card); border:1px solid var(--line); border-radius:12px; padding:22px; margin-top:14px; max-width:560px;}
|
||||
.camp{margin-bottom:14px;}
|
||||
.camp label{display:block; font-size:var(--fs-sm); color:var(--muted); margin-bottom:6px;}
|
||||
.camp input, .camp textarea, .camp select{
|
||||
width:100%; font-family:var(--font-ui); font-size:var(--fs-md); color:var(--ink);
|
||||
background:var(--card2); border:1px solid var(--line); border-radius:8px; padding:9px 12px; min-height:40px;}
|
||||
.camp input.mono{font-family:var(--font-mono);}
|
||||
.grid2{display:grid; grid-template-columns:1fr 1fr; gap:12px;}
|
||||
.op-row{display:flex; align-items:center; justify-content:space-between; gap:10px; padding:10px 0; border-bottom:1px solid var(--line2);}
|
||||
.op-name{font-size:var(--fs-md); font-weight:600;}
|
||||
.op-name small{font-weight:400; color:var(--muted); font-size:var(--fs-sm);}
|
||||
.chip{display:inline-flex; align-items:center; gap:6px; font-family:var(--font-mono); font-size:var(--fs-sm);
|
||||
background:color-mix(in srgb,var(--accent) 18%,transparent); color:var(--accent); padding:5px 10px; border-radius:7px;}
|
||||
.chip button{background:none; border:none; color:inherit; cursor:pointer; font-size:var(--fs-md); line-height:1;}
|
||||
.addcode{font-size:var(--fs-sm); border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line));
|
||||
background:transparent; color:var(--accent); border-radius:7px; padding:6px 12px; cursor:pointer;}
|
||||
.btn-primary{font-family:var(--font-ui); font-size:var(--fs-md); font-weight:600; height:42px; padding:0 20px;
|
||||
background:var(--accent); color:#fff; border:none; border-radius:8px; cursor:pointer;}
|
||||
.btn-ghost{font-family:var(--font-ui); font-size:var(--fs-md); height:42px; padding:0 18px;
|
||||
background:transparent; color:var(--ink); border:1px solid var(--line); border-radius:8px; cursor:pointer;}
|
||||
|
||||
/* tabel scala — referinta rapida */
|
||||
table.scala{width:100%; border-collapse:collapse; font-size:var(--fs-sm); margin-top:8px;}
|
||||
table.scala td{padding:7px 10px; border-bottom:1px solid var(--line2);}
|
||||
table.scala td:first-child{font-family:var(--font-mono); color:var(--accent); white-space:nowrap;}
|
||||
</style>
|
||||
</head>
|
||||
<body data-theme="grafit">
|
||||
<div class="wrap">
|
||||
<div class="toolbar">
|
||||
<button onclick="document.body.setAttribute('data-theme', document.body.getAttribute('data-theme')==='grafit'?'hartie':'grafit')">Comuta tema (grafit / hartie)</button>
|
||||
<span class="note">Fonturi: <span class="mono">system-ui, -apple-system, Segoe UI, Roboto…</span> — zero fisiere descarcate.</span>
|
||||
</div>
|
||||
|
||||
<h1>Gateway RAR AUTOPASS</h1>
|
||||
<p class="lead">Preview tipografie 5.16 — font stack nativ + scala uniforma, carduri aerisite, text mai mare.</p>
|
||||
|
||||
<div class="sec">Scala tipografica unica (tokeni)</div>
|
||||
<table class="scala">
|
||||
<tr><td>--fs-xs 12px</td><td style="font-size:var(--fs-xs)">Meta, hint-uri, sub-linii mono (azi 10px — prea mic)</td></tr>
|
||||
<tr><td>--fs-sm 13.5px</td><td style="font-size:var(--fs-sm)">Label-uri formular, pill-uri de stare (azi 11px)</td></tr>
|
||||
<tr><td>--fs-base 15px</td><td style="font-size:var(--fs-base)">Text body implicit pe toate paginile</td></tr>
|
||||
<tr><td>--fs-md 16px</td><td style="font-size:var(--fs-md)">Input-uri, VIN mono, text de card (azi 13px)</td></tr>
|
||||
<tr><td>--fs-2xl 28px</td><td style="font-size:var(--fs-2xl);font-weight:700">Cifra contor (azi 22px)</td></tr>
|
||||
</table>
|
||||
|
||||
<div class="sec">Dashboard — strip sanatate (DOT, nu bifa) + carduri-contor</div>
|
||||
<div class="strip">
|
||||
<span class="strip-left"><span class="dot live"></span> RAR online · declaratiile curg normal</span>
|
||||
<span class="strip-right">Ultima autentificare RAR: 28.06.2026 09:41</span>
|
||||
</div>
|
||||
<div class="contoare">
|
||||
<div class="contor-card"><div class="contor-cifra s-ok">847</div><div class="contor-label">Trimise (total)</div><div class="contor-sub">luna 124 · azi 9</div></div>
|
||||
<div class="contor-card"><div class="contor-cifra s-acc">12</div><div class="contor-label">In coada</div></div>
|
||||
<div class="contor-card"><div class="contor-cifra s-muted">0</div><div class="contor-label">De corectat</div></div>
|
||||
</div>
|
||||
|
||||
<div class="sec">Lista trimiteri — rand slim</div>
|
||||
<div class="lista">
|
||||
<div class="rand"><div><div class="slim-vin">WBA8E9...K7F2</div><div class="slim-meta">Inspectie tehnica · 09:42</div></div><span class="pill sent"><span class="pdot"></span>Trimis</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">WVWZZZ...3M1</div><div class="slim-meta">Revizie periodica · 09:38</div></div><span class="pill coada"><span class="pdot"></span>In coada</span></div>
|
||||
<div class="rand"><div><div class="slim-vin">VF1RFB...A88</div><div class="slim-meta">Sistem franare · 09:31</div></div><span class="pill err"><span class="pdot"></span>De corectat</span></div>
|
||||
</div>
|
||||
|
||||
<div class="sec">Formular editare — denumiri operatii in picker + adaugare operatie</div>
|
||||
<div class="form-card">
|
||||
<div class="camp"><label>VIN (serie sasiu)</label><input class="mono" value="WBA8E9C5K7F20143"></div>
|
||||
<div class="grid2">
|
||||
<div class="camp"><label>Data prestatiei</label><input class="mono" value="2026-06-22"></div>
|
||||
<div class="camp"><label>Numar inmatriculare</label><input class="mono" value="CT88NOE"></div>
|
||||
</div>
|
||||
<div class="camp"><label>Observatii (operatiile efectuate)</label><textarea rows="2">Revizie; schimbare placute frana</textarea></div>
|
||||
|
||||
<div class="camp">
|
||||
<label>Prestatii — cod RAR pe fiecare operatie</label>
|
||||
<div class="op-row">
|
||||
<span class="op-name">REVIZIE PERIODICA <small>— revizie la 15.000 km</small></span>
|
||||
<span style="display:flex;gap:8px;align-items:center;"><span class="chip">REV2 <button>×</button></span></span>
|
||||
</div>
|
||||
<div class="op-row" style="border-left:2px solid var(--warn); padding-left:10px;">
|
||||
<span class="op-name">SCHIMB PLACUTE FRANA <small style="color:var(--warn)">— lipsa cod</small></span>
|
||||
<select><option>— alege cod RAR —</option><option>FRN1 — Sistem de franare</option><option>REV2 — Revizie periodica</option></select>
|
||||
</div>
|
||||
<div style="margin-top:10px;"><button class="addcode">+ Adauga alta operatie / cod RAR</button></div>
|
||||
<p class="note">Picker-ul arata <strong>cod + denumire</strong> (FRN1 — Sistem de franare), nu doar codul.</p>
|
||||
</div>
|
||||
|
||||
<div style="display:flex; gap:10px; margin-top:18px;">
|
||||
<button class="btn-primary">Salveaza si retrimite</button>
|
||||
<button class="btn-ghost">Renunta</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p class="note" style="margin-top:30px;">Nota: tema/culorile sunt doar context. Subiectul acestui preview e <strong>fontul</strong> (system-ui) si <strong>scala</strong> (dimensiuni mai mari, uniforme). Deschide pe Windows si pe Mac ca sa vezi cum cade fontul nativ pe fiecare.</p>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
173
docs/mockups/prd-5.16-header-login-tema.html
Normal file
173
docs/mockups/prd-5.16-header-login-tema.html
Normal file
@@ -0,0 +1,173 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>PRD 5.16 — Header profesional + /login + selector tema stil landing</title>
|
||||
<style>
|
||||
:root{
|
||||
--font-ui: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
||||
--font-mono: ui-monospace, "SF Mono", "Cascadia Code", "Segoe UI Mono", "Roboto Mono", Menlo, Consolas, monospace;
|
||||
--fs-xs:12px; --fs-sm:13.5px; --fs-base:15px; --fs-md:16px; --fs-lg:18px; --fs-xl:20px; --fs-2xl:28px;
|
||||
--bg:#0f1218; --card:#181c24; --card2:#0f1218; --ink:#e6e9ef; --muted:#8b93a7;
|
||||
--line:#262b36; --line2:#1f2530; --accent:#6ea2ec; --ok:#2FBF8F; --warn:#E0A93B; --err:#E05D5D;
|
||||
--hbg:rgba(15,18,24,.88);
|
||||
}
|
||||
body[data-theme="hartie"]{
|
||||
--bg:#f3efe6; --card:#fffdf7; --card2:#f3efe6; --ink:#1e1a13; --muted:#6a6052;
|
||||
--line:#e2dccc; --line2:#ece6d9; --accent:#1F5FBF; --ok:#1c7d5d; --warn:#b45309; --err:#bd463c;
|
||||
--hbg:rgba(255,253,247,.9);
|
||||
}
|
||||
body[data-theme="cobalt"]{ --bg:#080d1c; --card:#111a33; --card2:#0b1226; --ink:#e9ecfb; --muted:#8a93b8; --line:#1d2747; --line2:#161f3a; --accent:#8aa0ff; --ok:#2fd0a6; --err:#f06a7a; --hbg:rgba(8,13,28,.9); }
|
||||
body[data-theme="cupru"]{ --bg:#15110b; --card:#211a12; --card2:#15110b; --ink:#efe6d6; --muted:#a89a85; --line:#36291c; --line2:#281e14; --accent:#dfa45c; --ok:#67b98c; --err:#e2685a; --hbg:rgba(21,17,11,.9); }
|
||||
*{box-sizing:border-box;}
|
||||
body{margin:0; background:var(--bg); color:var(--ink); font-family:var(--font-ui); font-size:var(--fs-base); -webkit-font-smoothing:antialiased;}
|
||||
.mono{font-family:var(--font-mono);}
|
||||
.muted{color:var(--muted);}
|
||||
|
||||
/* ===== HEADER aplicatie (logat) — profesional, branded ===== */
|
||||
header{
|
||||
display:grid; grid-template-columns:1fr auto 1fr; align-items:center;
|
||||
gap:16px; height:64px; padding:0 22px; background:var(--card); border:1px solid var(--line); border-radius:12px;
|
||||
}
|
||||
/* antet MINIMAL pe /login (neautentificat): doar logo + titlu + tema */
|
||||
.login-topbar{display:flex; align-items:center; justify-content:space-between; gap:16px; height:60px; padding:0 22px; background:var(--card); border:1px solid var(--line); border-radius:12px 12px 0 0; border-bottom:none;}
|
||||
.login-topbar .lt-brand{display:flex; align-items:center; gap:10px; font-weight:700; font-size:var(--fs-md);}
|
||||
.login-topbar .lt-brand .accent{color:var(--accent);}
|
||||
.h-left{display:flex; align-items:center; gap:12px;}
|
||||
.logo{height:32px; width:auto; display:block;}
|
||||
/* wordmark fallback in mockup (in app: PNG real ROMFAST) */
|
||||
.logo-fallback{display:inline-flex; align-items:center; gap:7px; font-weight:800; letter-spacing:-.01em; font-size:var(--fs-lg);}
|
||||
.logo-fallback .rom{color:#D1342F;} .logo-fallback .fast{color:var(--accent);}
|
||||
.h-center{text-align:center; line-height:1.15;}
|
||||
.h-title{font-size:var(--fs-md); font-weight:700; letter-spacing:.01em;}
|
||||
.h-title .accent{color:var(--accent);}
|
||||
.h-sub{font-size:var(--fs-xs); color:var(--muted); margin-top:2px;}
|
||||
.h-sub .svc{color:var(--ink); font-weight:600;}
|
||||
.env{display:inline-block; margin-left:8px; padding:1px 7px; border-radius:99px; font-size:10px; font-weight:700;
|
||||
text-transform:uppercase; letter-spacing:.04em; color:var(--warn); background:color-mix(in srgb,var(--warn) 16%,transparent);}
|
||||
.tier{display:inline-block; margin-left:6px; padding:1px 8px; border-radius:99px; font-size:10px; font-weight:700; text-transform:uppercase; letter-spacing:.04em; color:var(--accent); background:color-mix(in srgb,var(--accent) 16%,transparent);}
|
||||
.h-right{display:flex; align-items:center; justify-content:flex-end; gap:10px;}
|
||||
.rar-chip{display:flex; align-items:center; gap:8px; height:38px; padding:0 13px; border-radius:99px; border:1px solid color-mix(in srgb,var(--ok) 35%,var(--line)); background:color-mix(in srgb,var(--ok) 10%,transparent); color:var(--ok); font-size:var(--fs-sm); font-weight:600; cursor:default;}
|
||||
.rar-chip .dot{width:9px; height:9px; border-radius:99px; background:currentColor; box-shadow:0 0 0 4px color-mix(in srgb,currentColor 22%,transparent);}
|
||||
|
||||
/* selector tema STIL LANDING: pill cu icon + eticheta tema curenta */
|
||||
.tema-btn{display:flex; align-items:center; gap:8px; height:38px; padding:0 13px; border-radius:8px;
|
||||
background:transparent; border:1px solid var(--line); color:var(--muted); font-family:var(--font-ui);
|
||||
font-size:var(--fs-sm); cursor:pointer; transition:border-color .15s, color .15s;}
|
||||
.tema-btn:hover{border-color:var(--accent); color:var(--ink);}
|
||||
.tema-btn svg{flex-shrink:0;}
|
||||
.ver{font-size:var(--fs-xs); color:var(--muted);}
|
||||
.icon-btn{width:38px; height:38px; border-radius:8px; border:1px solid var(--line); background:transparent;
|
||||
color:var(--ink); font-size:18px; cursor:pointer; display:inline-flex; align-items:center; justify-content:center;}
|
||||
|
||||
.wrap{max-width:1100px; margin:0 auto; padding:24px 22px 60px;}
|
||||
.sec{font-size:var(--fs-lg); margin:30px 0 12px; padding-bottom:6px; border-bottom:1px solid var(--line);}
|
||||
.note{font-size:var(--fs-sm); color:var(--muted);}
|
||||
.toolbar{display:flex; gap:10px; align-items:center; margin:14px 0;}
|
||||
.toolbar button{font-family:var(--font-ui); font-size:var(--fs-sm); height:34px; padding:0 12px; border-radius:7px; border:1px solid var(--line); background:var(--card); color:var(--ink); cursor:pointer;}
|
||||
|
||||
/* ===== /login profesional ===== */
|
||||
.login-shell{min-height:520px; display:grid; grid-template-columns:1.1fr .9fr; border:1px solid var(--line); border-radius:16px; overflow:hidden; background:var(--card);}
|
||||
.login-aside{padding:40px 38px; background:linear-gradient(160deg, color-mix(in srgb,var(--accent) 14%,var(--card)), var(--card)); border-right:1px solid var(--line); display:flex; flex-direction:column; justify-content:center;}
|
||||
.login-brand{display:flex; align-items:center; gap:10px; margin-bottom:22px;}
|
||||
.login-brand .logo-fallback{font-size:var(--fs-xl);}
|
||||
.login-aside h2{font-size:var(--fs-2xl); line-height:1.2; margin:0 0 12px; letter-spacing:-.02em;}
|
||||
.login-aside p{font-size:var(--fs-md); color:var(--muted); line-height:1.6; margin:0 0 18px; max-width:380px;}
|
||||
.trust{display:flex; flex-direction:column; gap:9px; margin-top:6px;}
|
||||
.trust div{display:flex; align-items:center; gap:9px; font-size:var(--fs-sm); color:var(--ink);}
|
||||
.trust svg{flex-shrink:0; color:var(--ok);}
|
||||
.login-form{padding:40px 38px; display:flex; flex-direction:column; justify-content:center;}
|
||||
.login-form h3{font-size:var(--fs-xl); margin:0 0 4px;}
|
||||
.login-form .lead{font-size:var(--fs-sm); color:var(--muted); margin:0 0 22px;}
|
||||
.field{margin-bottom:16px;}
|
||||
.field label{display:block; font-size:var(--fs-sm); color:var(--muted); margin-bottom:6px;}
|
||||
.field input{width:100%; font-family:var(--font-ui); font-size:var(--fs-md); color:var(--ink); background:var(--card2); border:1px solid var(--line); border-radius:8px; padding:11px 13px; min-height:44px;}
|
||||
.field input:focus{outline:2px solid var(--accent); border-color:var(--accent);}
|
||||
.btn-primary{width:100%; height:46px; font-family:var(--font-ui); font-size:var(--fs-md); font-weight:600; background:var(--accent); color:#fff; border:none; border-radius:8px; cursor:pointer; margin-top:4px;}
|
||||
.row-between{display:flex; align-items:center; justify-content:space-between; margin:-4px 0 18px;}
|
||||
.link{color:var(--accent); font-size:var(--fs-sm); text-decoration:none;}
|
||||
.login-foot{text-align:center; font-size:var(--fs-sm); color:var(--muted); margin-top:18px;}
|
||||
</style>
|
||||
</head>
|
||||
<body data-theme="grafit">
|
||||
|
||||
<div class="wrap">
|
||||
<div class="toolbar">
|
||||
<span class="note">Comuta tema cu butonul de tema (stil landing: icon + eticheta).</span>
|
||||
</div>
|
||||
|
||||
<!-- ===== A. Antet aplicatie — LOGAT ===== -->
|
||||
<div class="sec">Antet aplicatie — LOGAT (branded)</div>
|
||||
<header>
|
||||
<div class="h-left">
|
||||
<span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span>
|
||||
<span class="note" style="font-size:var(--fs-xs)">(in app: PNG logo real)</span>
|
||||
</div>
|
||||
<div class="h-center">
|
||||
<div class="h-title">ROMFAST <span class="accent">AUTOPASS</span><span class="env">test</span><span class="tier">Pro</span></div>
|
||||
<div class="h-sub">Service auto: <span class="svc">Service Auto Vâlcea SRL</span></div>
|
||||
</div>
|
||||
<div class="h-right">
|
||||
<div class="rar-chip" title="Ultima autentificare RAR: 28.06.2026 09:41"><span class="dot"></span> RAR online</div>
|
||||
<button class="tema-btn" onclick="cycle()">
|
||||
<svg id="t-ic" width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg>
|
||||
<span id="t-label">Grafit</span>
|
||||
</button>
|
||||
<span class="ver">v5.16</span>
|
||||
<button class="icon-btn" title="Meniu cont">☰</button>
|
||||
</div>
|
||||
</header>
|
||||
<p class="note">Doar cand esti LOGAT: titlu <strong>ROMFAST AUTOPASS</strong> + badge plan
|
||||
(<span class="mono">accounts.tier</span>) + sub titlu numele service-ului (<span class="mono">accounts.name</span>);
|
||||
dreapta dot <strong>RAR online</strong> + selector tema + meniu cont. Toate gate-uite pe
|
||||
<span class="mono">is_authenticated</span>.</p>
|
||||
|
||||
<!-- ===== B. /login — NEAUTENTIFICAT (antet minimal) ===== -->
|
||||
<div class="sec">Pagina /login — NEAUTENTIFICAT (antet minimal)</div>
|
||||
<div class="login-topbar">
|
||||
<span class="lt-brand"><span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span> ROMFAST <span class="accent">AUTOPASS</span></span>
|
||||
<button class="tema-btn" onclick="cycle()">
|
||||
<svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg>
|
||||
<span id="t-label2">Grafit</span>
|
||||
</button>
|
||||
</div>
|
||||
<div class="login-shell" style="border-radius:0 0 16px 16px; border-top:none;">
|
||||
<div class="login-aside">
|
||||
<div class="login-brand"><span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span></div>
|
||||
<h2>ROMFAST <span style="color:var(--accent)">AUTOPASS</span></h2>
|
||||
<p>Declară prestațiile de service-auto la RAR AUTOPASS, automat. Conform Legii 142/2023.</p>
|
||||
<div class="trust">
|
||||
<div><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.2"><path d="M20 6L9 17l-5-5"/></svg> Conform Legii 142/2023 și OMTI 210/2024</div>
|
||||
<div><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.8"><rect x="5" y="11" width="14" height="9" rx="1.5"/><path d="M8 11V8a4 4 0 0 1 8 0v3"/></svg> Datele tale criptate, șterse la 3 luni</div>
|
||||
<div><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M20 6L9 17l-5-5"/></svg> Parte din familia ROA — Romfast Applications</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="login-form">
|
||||
<h3>Autentificare</h3>
|
||||
<p class="lead">Intră în contul service-ului tău.</p>
|
||||
<div class="field"><label>Email</label><input type="email" value="contact@service-valcea.ro"></div>
|
||||
<div class="field"><label>Parolă</label><input type="password" value="••••••••••"></div>
|
||||
<div class="row-between"><span></span><a class="link" href="#">Ai uitat parola?</a></div>
|
||||
<button class="btn-primary">Intră în cont</button>
|
||||
<div class="login-foot">Cont nou? <a class="link" href="/signup">Înregistrează service-ul</a></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p class="note">Antetul de <span class="mono">/login</span> NU are dot RAR, nume service sau badge plan —
|
||||
utilizatorul nu e logat inca. Doar logo + titlu <strong>ROMFAST AUTOPASS</strong> + selector tema.
|
||||
(RAR/service/plan/meniu apar abia dupa autentificare.)</p>
|
||||
|
||||
<div class="sec">Landing — butonul „Autentificare" duce la /login</div>
|
||||
<p class="note">Pe landing, „Autentificare" (azi deschide modalul de register din landing pe tab-ul
|
||||
login) devine un link real către <span class="mono">/login</span> (pagina de mai sus). „Creează cont"
|
||||
rămâne neschimbat. Selectorul de teme din landing e exact modelul pe care îl preia aplicația.</p>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
var THEMES=[['grafit','Grafit'],['cobalt','Cobalt'],['cupru','Cupru'],['hartie','Hârtie']];
|
||||
var i=0;
|
||||
function cycle(){ i=(i+1)%THEMES.length; document.body.setAttribute('data-theme',THEMES[i][0]); document.getElementById('t-label').textContent=THEMES[i][1]; var l2=document.getElementById('t-label2'); if(l2)l2.textContent=THEMES[i][1]; }
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
275
docs/mockups/prd-5.16-import-wizard.html
Normal file
275
docs/mockups/prd-5.16-import-wizard.html
Normal file
@@ -0,0 +1,275 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ro">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>PRD 5.16 — Wizard import fișier (4 pași) + editare/corecție</title>
|
||||
<style>
|
||||
:root{
|
||||
--font-ui: system-ui, -apple-system, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
||||
--font-mono: ui-monospace, "SF Mono", "Cascadia Code", "Segoe UI Mono", "Roboto Mono", Menlo, Consolas, monospace;
|
||||
--fs-xs:12px; --fs-sm:13.5px; --fs-base:15px; --fs-md:16px; --fs-lg:18px; --fs-xl:20px; --fs-2xl:28px;
|
||||
--bg:#0f1218; --card:#181c24; --card2:#0f1218; --ink:#e6e9ef; --muted:#8b93a7;
|
||||
--line:#262b36; --line2:#1f2530; --accent:#6ea2ec; --ok:#2FBF8F; --warn:#E0A93B; --err:#E05D5D;
|
||||
--hbg:rgba(15,18,24,.9);
|
||||
}
|
||||
body[data-theme="hartie"]{ --bg:#f3efe6; --card:#fffdf7; --card2:#f3efe6; --ink:#1e1a13; --muted:#6a6052; --line:#e2dccc; --line2:#ece6d9; --accent:#1F5FBF; --ok:#1c7d5d; --warn:#b45309; --err:#bd463c; --hbg:rgba(255,253,247,.92); }
|
||||
*{box-sizing:border-box;}
|
||||
body{margin:0; background:var(--bg); color:var(--ink); font-family:var(--font-ui); font-size:var(--fs-base); -webkit-font-smoothing:antialiased;}
|
||||
.mono{font-family:var(--font-mono);} .muted{color:var(--muted);}
|
||||
|
||||
header{position:sticky; top:0; z-index:5; display:grid; grid-template-columns:1fr auto 1fr; align-items:center; gap:16px; height:64px; padding:0 22px; background:var(--hbg); backdrop-filter:blur(8px); border-bottom:1px solid var(--line);}
|
||||
.logo-fallback{display:inline-flex; align-items:center; gap:6px; font-weight:800; font-size:var(--fs-lg);}
|
||||
.logo-fallback .rom{color:#D1342F;} .logo-fallback .fast{color:var(--accent);}
|
||||
.h-center{text-align:center; line-height:1.15;}
|
||||
.h-title{font-size:var(--fs-md); font-weight:700;} .h-title .accent{color:var(--accent);}
|
||||
.h-sub{font-size:var(--fs-xs); color:var(--muted); margin-top:2px;} .h-sub .svc{color:var(--ink); font-weight:600;}
|
||||
.env{display:inline-block; margin-left:8px; padding:1px 7px; border-radius:99px; font-size:10px; font-weight:700; text-transform:uppercase; color:var(--warn); background:color-mix(in srgb,var(--warn) 16%,transparent);}
|
||||
.tier{display:inline-block; margin-left:6px; padding:1px 8px; border-radius:99px; font-size:10px; font-weight:700; text-transform:uppercase; letter-spacing:.04em; color:var(--accent); background:color-mix(in srgb,var(--accent) 16%,transparent);}
|
||||
.h-right{display:flex; align-items:center; justify-content:flex-end; gap:10px;}
|
||||
.rar-chip{display:flex; align-items:center; gap:8px; height:38px; padding:0 13px; border-radius:99px; border:1px solid color-mix(in srgb,var(--ok) 35%,var(--line)); background:color-mix(in srgb,var(--ok) 10%,transparent); color:var(--ok); font-size:var(--fs-sm); font-weight:600; cursor:default;}
|
||||
.rar-chip .dot{width:9px; height:9px; border-radius:99px; background:currentColor; box-shadow:0 0 0 4px color-mix(in srgb,currentColor 22%,transparent);}
|
||||
.tema-btn{display:flex; align-items:center; gap:8px; height:38px; padding:0 13px; border-radius:8px; background:transparent; border:1px solid var(--line); color:var(--muted); font-size:var(--fs-sm); cursor:pointer;}
|
||||
.icon-btn{width:38px; height:38px; border-radius:8px; border:1px solid var(--line); background:transparent; color:var(--ink); font-size:18px; cursor:pointer; display:inline-flex; align-items:center; justify-content:center;}
|
||||
|
||||
.wrap{max-width:1000px; margin:0 auto; padding:22px 22px 70px;}
|
||||
.screen-cap{font-size:var(--fs-xs); text-transform:uppercase; letter-spacing:.08em; color:var(--accent); font-weight:700; margin:30px 0 10px;}
|
||||
|
||||
/* stepper slim */
|
||||
.stepper{display:flex; align-items:center; gap:0; background:var(--card); border:1px solid var(--line); border-radius:11px; padding:6px; margin-bottom:14px;}
|
||||
.step{flex:1; display:flex; align-items:center; gap:9px; padding:9px 12px; border-radius:8px; font-size:var(--fs-sm);}
|
||||
.step .num{display:inline-flex; width:24px; height:24px; align-items:center; justify-content:center; border-radius:99px; font-size:var(--fs-sm); font-weight:700; background:var(--card2); border:1px solid var(--line); color:var(--muted); flex-shrink:0;}
|
||||
.step.done .num{background:color-mix(in srgb,var(--ok) 20%,transparent); border-color:transparent; color:var(--ok);}
|
||||
.step.active{background:color-mix(in srgb,var(--accent) 14%,transparent);}
|
||||
.step.active .num{background:var(--accent); border-color:transparent; color:#fff;}
|
||||
.step.active .t{color:var(--ink); font-weight:600;} .step .t{color:var(--muted);}
|
||||
.step .sep{color:var(--line);}
|
||||
|
||||
.panel{background:var(--card); border:1px solid var(--line); border-radius:12px; overflow:hidden;}
|
||||
.panel-head{padding:16px 18px; border-bottom:1px solid var(--line);}
|
||||
.panel-head h3{margin:0; font-size:var(--fs-lg);}
|
||||
.panel-head p{margin:4px 0 0; font-size:var(--fs-sm); color:var(--muted);}
|
||||
.panel-body{padding:18px;}
|
||||
.foot{display:flex; align-items:center; justify-content:space-between; gap:12px; padding:14px 18px; border-top:1px solid var(--line); background:var(--card2);}
|
||||
.btn-primary{font-size:var(--fs-md); font-weight:600; height:44px; padding:0 22px; background:var(--accent); color:#fff; border:none; border-radius:9px; cursor:pointer;}
|
||||
.btn-ghost{font-size:var(--fs-md); height:44px; padding:0 18px; background:transparent; color:var(--ink); border:1px solid var(--line); border-radius:9px; cursor:pointer;}
|
||||
|
||||
/* PAS 1 — drop zone */
|
||||
.drop{border:2px dashed color-mix(in srgb,var(--accent) 45%,var(--line)); border-radius:12px; padding:46px 20px; text-align:center; background:var(--card2);}
|
||||
.drop .ic{width:54px; height:54px; border-radius:12px; margin:0 auto 14px; display:flex; align-items:center; justify-content:center; background:color-mix(in srgb,var(--accent) 14%,transparent); color:var(--accent);}
|
||||
.drop .big{font-size:var(--fs-lg); font-weight:700;}
|
||||
.drop .sm{font-size:var(--fs-sm); color:var(--muted); margin:6px 0 16px;}
|
||||
.formate{display:inline-flex; gap:8px; margin-top:14px;}
|
||||
.badge-fmt{font-family:var(--font-mono); font-size:var(--fs-xs); padding:3px 9px; border-radius:6px; background:var(--card); border:1px solid var(--line); color:var(--muted);}
|
||||
|
||||
/* PAS 2 — mapare coloane */
|
||||
.memo{display:flex; align-items:center; gap:9px; font-size:var(--fs-sm); color:var(--ok); background:color-mix(in srgb,var(--ok) 12%,transparent); border:1px solid color-mix(in srgb,var(--ok) 28%,transparent); border-radius:9px; padding:10px 14px; margin-bottom:14px;}
|
||||
table{width:100%; border-collapse:collapse; font-size:var(--fs-base);}
|
||||
.map th{text-align:left; font-size:var(--fs-xs); text-transform:uppercase; letter-spacing:.05em; color:var(--muted); padding:0 12px 8px; font-weight:700;}
|
||||
.map td{padding:9px 12px; border-top:1px solid var(--line2); vertical-align:middle;}
|
||||
.col-name{font-family:var(--font-mono); font-size:var(--fs-sm); font-weight:600;}
|
||||
.col-sample{font-family:var(--font-mono); font-size:var(--fs-xs); color:var(--muted);}
|
||||
.map select{width:100%; font-family:var(--font-ui); font-size:var(--fs-base); color:var(--ink); background:var(--card2); border:1px solid var(--line); border-radius:8px; padding:8px 10px; min-height:38px;}
|
||||
.map .ignored select{color:var(--muted);}
|
||||
.switch{display:inline-flex; align-items:center; gap:9px; font-size:var(--fs-sm); color:var(--muted);}
|
||||
.switch .track{width:38px; height:22px; border-radius:99px; background:color-mix(in srgb,var(--accent) 70%,var(--line)); position:relative;}
|
||||
.switch .knob{position:absolute; top:2px; right:2px; width:18px; height:18px; border-radius:99px; background:#fff;}
|
||||
|
||||
/* PAS 3 — preview */
|
||||
.summary{display:flex; gap:10px; flex-wrap:wrap; margin-bottom:14px;}
|
||||
.chipc{display:flex; align-items:center; gap:8px; font-size:var(--fs-sm); padding:7px 13px; border-radius:99px; border:1px solid var(--line); background:var(--card2);}
|
||||
.chipc b{font-size:var(--fs-md);}
|
||||
.pv th{text-align:left; font-size:var(--fs-xs); text-transform:uppercase; letter-spacing:.05em; color:var(--muted); padding:0 12px 9px; font-weight:700;}
|
||||
.pv td{padding:11px 12px; border-top:1px solid var(--line2); font-size:var(--fs-sm);}
|
||||
.pv .vin{font-family:var(--font-mono); font-size:var(--fs-sm);}
|
||||
.pill{display:inline-flex; align-items:center; gap:6px; padding:4px 11px; border-radius:99px; font-size:var(--fs-xs); font-weight:600;}
|
||||
.pill .pdot{width:7px; height:7px; border-radius:99px;}
|
||||
.ok{background:color-mix(in srgb,var(--ok) 14%,transparent); color:var(--ok);} .ok .pdot{background:var(--ok);}
|
||||
.warn{background:color-mix(in srgb,var(--warn) 16%,transparent); color:var(--warn);} .warn .pdot{background:var(--warn);}
|
||||
.err{background:color-mix(in srgb,var(--err) 14%,transparent); color:var(--err);} .err .pdot{background:var(--err);}
|
||||
.lnk{color:var(--accent); font-size:var(--fs-sm); cursor:pointer; background:none; border:none; padding:0; text-decoration:underline;}
|
||||
tr.editing{background:color-mix(in srgb,var(--accent) 7%,transparent);}
|
||||
|
||||
/* editare inline / corectie (slim form) */
|
||||
.editbox{margin:2px 12px 12px; border:1px solid color-mix(in srgb,var(--accent) 35%,var(--line)); border-radius:11px; background:var(--card2); padding:16px;}
|
||||
.editbox .et{font-size:var(--fs-sm); font-weight:700; margin-bottom:12px; color:var(--accent);}
|
||||
.field{margin-bottom:13px;}
|
||||
.field label{display:block; font-size:var(--fs-sm); color:var(--muted); margin-bottom:6px;}
|
||||
.field input, .field textarea, .field select{width:100%; font-family:var(--font-ui); font-size:var(--fs-md); color:var(--ink); background:var(--card); border:1px solid var(--line); border-radius:8px; padding:9px 12px; min-height:40px;}
|
||||
.field input.mono{font-family:var(--font-mono);}
|
||||
.grid3{display:grid; grid-template-columns:1.3fr 1fr 1fr; gap:12px;}
|
||||
.op-row{display:flex; align-items:center; justify-content:space-between; gap:10px; padding:9px 0; border-bottom:1px solid var(--line2);}
|
||||
.op-name{font-size:var(--fs-md); font-weight:600;} .op-name small{font-weight:400; color:var(--muted); font-size:var(--fs-sm);}
|
||||
.chip{display:inline-flex; align-items:center; gap:6px; font-family:var(--font-mono); font-size:var(--fs-sm); background:color-mix(in srgb,var(--accent) 18%,transparent); color:var(--accent); padding:5px 10px; border-radius:7px;}
|
||||
.chip button{background:none; border:none; color:inherit; cursor:pointer; font-size:var(--fs-md);}
|
||||
.addcode{font-size:var(--fs-sm); border:1px dashed color-mix(in srgb,var(--accent) 55%,var(--line)); background:transparent; color:var(--accent); border-radius:7px; padding:6px 12px; cursor:pointer;}
|
||||
.save-rule{font-size:var(--fs-xs); color:var(--muted); text-decoration:underline; background:none; border:none; cursor:pointer;}
|
||||
.actrow{display:flex; gap:10px; margin-top:14px;}
|
||||
|
||||
/* PAS 4 — confirma */
|
||||
.confirm-big{text-align:center; padding:8px 0 4px;}
|
||||
.confirm-big .n{font-size:42px; font-weight:700; color:var(--ok); line-height:1;}
|
||||
.confirm-big .l{font-size:var(--fs-md); color:var(--muted); margin-top:6px;}
|
||||
.breakdown{display:flex; gap:10px; justify-content:center; margin:16px 0;}
|
||||
.atest{display:flex; align-items:flex-start; gap:10px; font-size:var(--fs-sm); color:var(--ink); background:var(--card2); border:1px solid var(--line); border-radius:10px; padding:14px 16px; margin-top:6px;}
|
||||
.atest input{margin-top:3px; width:18px; height:18px;}
|
||||
.warn-note{display:flex; align-items:center; gap:9px; font-size:var(--fs-sm); color:var(--warn); margin-top:12px;}
|
||||
</style>
|
||||
</head>
|
||||
<body data-theme="grafit">
|
||||
|
||||
<header>
|
||||
<span class="logo-fallback"><span class="rom">ROM</span><span class="fast">FAST</span></span>
|
||||
<div class="h-center">
|
||||
<div class="h-title">ROMFAST <span class="accent">AUTOPASS</span><span class="env">test</span><span class="tier">Pro</span></div>
|
||||
<div class="h-sub">Service auto: <span class="svc">Service Auto Vâlcea SRL</span></div>
|
||||
</div>
|
||||
<div class="h-right">
|
||||
<div class="rar-chip" title="Ultima autentificare RAR: 28.06.2026 09:41"><span class="dot"></span> RAR online</div>
|
||||
<button class="tema-btn"><svg width="17" height="17" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.6"><circle cx="12" cy="12" r="9"/><path d="M12 3a9 9 0 0 0 0 18z" fill="currentColor" stroke="none"/></svg> Grafit</button>
|
||||
<button class="icon-btn">☰</button>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
<div class="wrap">
|
||||
|
||||
<!-- ============ PAS 1 ============ -->
|
||||
<div class="screen-cap">Pas 1 — Încarcă fișier</div>
|
||||
<div class="stepper">
|
||||
<div class="step active"><span class="num">1</span><span class="t">Încarcă</span></div>
|
||||
<div class="step"><span class="num">2</span><span class="t">Potrivește</span></div>
|
||||
<div class="step"><span class="num">3</span><span class="t">Verifică</span></div>
|
||||
<div class="step"><span class="num">4</span><span class="t">Confirmă</span></div>
|
||||
</div>
|
||||
<div class="panel">
|
||||
<div class="panel-head"><h3>Încarcă fișierul cu prestații</h3><p>Trage un fișier xlsx/csv aici sau folosește butonul de alegere.</p></div>
|
||||
<div class="panel-body">
|
||||
<div class="drop">
|
||||
<div class="ic"><svg width="26" height="26" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7"><path d="M12 16V4M7 9l5-5 5 5"/><path d="M5 20h14"/></svg></div>
|
||||
<div class="big">Trage fișierul aici</div>
|
||||
<div class="sm">sau apasă pentru a alege de pe calculator · max 5 MB</div>
|
||||
<button class="btn-primary">Alege fișier</button>
|
||||
<div class="formate"><span class="badge-fmt">.xlsx</span><span class="badge-fmt">.csv</span><span class="badge-fmt">.xls</span></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ============ PAS 2 ============ -->
|
||||
<div class="screen-cap">Pas 2 — Potrivește coloanele</div>
|
||||
<div class="stepper">
|
||||
<div class="step done"><span class="num">✓</span><span class="t">Încarcă</span></div>
|
||||
<div class="step active"><span class="num">2</span><span class="t">Potrivește</span></div>
|
||||
<div class="step"><span class="num">3</span><span class="t">Verifică</span></div>
|
||||
<div class="step"><span class="num">4</span><span class="t">Confirmă</span></div>
|
||||
</div>
|
||||
<div class="panel">
|
||||
<div class="panel-head"><h3>Potrivește coloanele fișierului cu câmpurile RAR</h3><p>Spune-ne ce coloană din fișier corespunde cu ce câmp RAR. <span class="mono">prestatii-iunie.xlsx</span> · 38 rânduri.</p></div>
|
||||
<div class="panel-body">
|
||||
<div class="memo"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M20 6L9 17l-5-5"/></svg> Format recunoscut — am reaplicat maparea salvată pentru aceste coloane.</div>
|
||||
<table class="map">
|
||||
<thead><tr><th style="width:34%">Coloană din fișier</th><th style="width:30%">Exemplu</th><th style="width:36%">Câmp RAR</th></tr></thead>
|
||||
<tbody>
|
||||
<tr><td class="col-name">SASIU</td><td class="col-sample">WBA8E9C5K7F20143</td><td><select><option>VIN (serie șasiu)</option></select></td></tr>
|
||||
<tr><td class="col-name">DATA</td><td class="col-sample">22.06.2026</td><td><select><option>Data prestației</option></select></td></tr>
|
||||
<tr><td class="col-name">NR_AUTO</td><td class="col-sample">CT88NOE</td><td><select><option>Număr înmatriculare</option></select></td></tr>
|
||||
<tr><td class="col-name">KM</td><td class="col-sample">142500</td><td><select><option>Odometru (km)</option></select></td></tr>
|
||||
<tr><td class="col-name">OPERATIE</td><td class="col-sample">Revizie periodică</td><td><select><option>Operație service → cod RAR</option></select></td></tr>
|
||||
<tr class="ignored"><td class="col-name">PRET</td><td class="col-sample">350 lei</td><td><select><option>— ignoră coloana —</option></select></td></tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
<div class="foot">
|
||||
<label class="switch"><span class="track"><span class="knob"></span></span> Ține minte maparea pentru acest format</label>
|
||||
<div style="display:flex; gap:10px;"><button class="btn-ghost">Înapoi</button><button class="btn-primary">Continuă spre verificare</button></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ============ PAS 3 ============ -->
|
||||
<div class="screen-cap">Pas 3 — Verifică (cu editare/corecție rând)</div>
|
||||
<div class="stepper">
|
||||
<div class="step done"><span class="num">✓</span><span class="t">Încarcă</span></div>
|
||||
<div class="step done"><span class="num">✓</span><span class="t">Potrivește</span></div>
|
||||
<div class="step active"><span class="num">3</span><span class="t">Verifică</span></div>
|
||||
<div class="step"><span class="num">4</span><span class="t">Confirmă</span></div>
|
||||
</div>
|
||||
<div class="panel">
|
||||
<div class="panel-head"><h3>Verifică rândurile înainte să le trimiți la RAR</h3><p>Corectează rândurile marcate. Restul sunt gata de trimis.</p></div>
|
||||
<div class="panel-body">
|
||||
<div class="summary">
|
||||
<span class="chipc"><span class="pill ok"><span class="pdot"></span></span> <b>33</b> gata</span>
|
||||
<span class="chipc"><span class="pill warn"><span class="pdot"></span></span> <b>2</b> Cod RAR lipsă</span>
|
||||
<span class="chipc"><span class="pill err"><span class="pdot"></span></span> <b>1</b> Date incomplete</span>
|
||||
<span class="chipc"><span class="pill warn"><span class="pdot"></span></span> <b>1</b> Duplicat în fișier</span>
|
||||
<span class="chipc"><span class="pill ok"><span class="pdot"></span></span> <b>1</b> Deja trimis</span>
|
||||
</div>
|
||||
<table class="pv">
|
||||
<thead><tr><th>VIN</th><th>Operație</th><th>Data</th><th>Stare</th><th></th></tr></thead>
|
||||
<tbody>
|
||||
<tr><td class="vin">WBA8E9...K7F2</td><td>Inspecție tehnică</td><td class="mono">22.06.2026</td><td><span class="pill ok"><span class="pdot"></span>Gata</span></td><td><button class="lnk">editează</button></td></tr>
|
||||
<!-- rand in editare/corectie -->
|
||||
<tr class="editing"><td class="vin">VF1RFB...A88</td><td>Schimb plăcuțe frână</td><td class="mono">22.06.2026</td><td><span class="pill warn"><span class="pdot"></span>Cod RAR lipsă</span></td><td><button class="lnk">închide</button></td></tr>
|
||||
<tr class="editing"><td colspan="5" style="padding:0;">
|
||||
<div class="editbox">
|
||||
<div class="et">Corectează rândul — VF1RFB...A88</div>
|
||||
<div class="grid3">
|
||||
<div class="field"><label>VIN (serie șasiu)</label><input class="mono" value="VF1RFB00A88142073"></div>
|
||||
<div class="field"><label>Data prestației</label><input class="mono" value="2026-06-22"></div>
|
||||
<div class="field"><label>Nr. înmatriculare</label><input class="mono" value="CT88NOE"></div>
|
||||
</div>
|
||||
<div class="field"><label>Observații (operațiile efectuate)</label><textarea rows="2">Schimbare plăcuțe frână față</textarea></div>
|
||||
<div class="field">
|
||||
<label>Prestații — cod RAR pe fiecare operație</label>
|
||||
<div class="op-row" style="border-left:2px solid var(--warn); padding-left:10px;">
|
||||
<span class="op-name">SCHIMB PLĂCUȚE FRÂNĂ <small style="color:var(--warn)">— lipsă cod</small></span>
|
||||
<span style="display:flex; gap:8px; align-items:center;">
|
||||
<select><option>— alege cod RAR —</option><option>FRN1 — Sistem de frânare</option><option>REV2 — Revizie periodică</option></select>
|
||||
</span>
|
||||
</div>
|
||||
<div style="margin-top:8px; display:flex; align-items:center; gap:12px;">
|
||||
<button class="addcode">+ Adaugă altă operație / cod RAR</button>
|
||||
<button class="save-rule">salvează ca regulă op→cod (deblochează rândurile la fel)</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="actrow"><button class="btn-primary">Salvează rândul</button><button class="btn-ghost">Renunță</button></div>
|
||||
</div>
|
||||
</td></tr>
|
||||
<tr><td class="vin">ZAR937...C04</td><td>Schimb ulei</td><td class="mono">21.06.2026</td><td><span class="pill err"><span class="pdot"></span>Date incomplete</span></td><td><button class="lnk">editează</button></td></tr>
|
||||
<tr><td class="vin">WVWZZZ...3M1</td><td>Revizie periodică</td><td class="mono">22.06.2026</td><td><span class="pill warn"><span class="pdot"></span>Duplicat în fișier</span></td><td><button class="lnk">editează</button></td></tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
<div class="foot">
|
||||
<span class="muted" style="font-size:var(--fs-sm);">3 rânduri de corectat înainte de trimitere</span>
|
||||
<div style="display:flex; gap:10px;"><button class="btn-ghost">Înapoi</button><button class="btn-primary">Confirmă valorile →</button></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ============ PAS 4 ============ -->
|
||||
<div class="screen-cap">Pas 4 — Confirmă trimiterea</div>
|
||||
<div class="stepper">
|
||||
<div class="step done"><span class="num">✓</span><span class="t">Încarcă</span></div>
|
||||
<div class="step done"><span class="num">✓</span><span class="t">Potrivește</span></div>
|
||||
<div class="step done"><span class="num">✓</span><span class="t">Verifică</span></div>
|
||||
<div class="step active"><span class="num">4</span><span class="t">Confirmă</span></div>
|
||||
</div>
|
||||
<div class="panel">
|
||||
<div class="panel-head"><h3>Confirmă trimiterea la RAR</h3><p>Acțiunea e ireversibilă — prestațiile pleacă la RAR AUTOPASS.</p></div>
|
||||
<div class="panel-body">
|
||||
<div class="confirm-big"><div class="n">36</div><div class="l">prestații gata de trimis</div></div>
|
||||
<div class="breakdown">
|
||||
<span class="chipc"><span class="pill ok"><span class="pdot"></span></span> 36 vor pleca</span>
|
||||
<span class="chipc"><span class="pill warn"><span class="pdot"></span></span> 1 sărit (duplicat)</span>
|
||||
<span class="chipc"><span class="pill ok"><span class="pdot"></span></span> 1 deja trimis</span>
|
||||
</div>
|
||||
<label class="atest"><input type="checkbox" checked> Confirm că datele sunt corecte și autorizez trimiterea celor 36 de prestații la RAR AUTOPASS, conform Legii 142/2023.</label>
|
||||
<div class="warn-note"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.8"><path d="M12 9v4M12 17h.01"/><path d="M10.3 3.8 2 18a2 2 0 0 0 1.7 3h16.6a2 2 0 0 0 1.7-3L13.7 3.8a2 2 0 0 0-3.4 0z"/></svg> O prestație finalizată la RAR nu mai poate fi anulată sau corectată prin aplicație.</div>
|
||||
</div>
|
||||
<div class="foot">
|
||||
<button class="btn-ghost">Înapoi la verificare</button>
|
||||
<button class="btn-primary">Trimite 36 de prestații la RAR</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
13171
docs/operatii-service/operatii-service-automotive.csv
Normal file
13171
docs/operatii-service/operatii-service-automotive.csv
Normal file
File diff suppressed because it is too large
Load Diff
1669
docs/operatii-service/operatii-service-clever.csv
Normal file
1669
docs/operatii-service/operatii-service-clever.csv
Normal file
File diff suppressed because it is too large
Load Diff
3744
docs/operatii-service/operatii-service-sigma.csv
Normal file
3744
docs/operatii-service/operatii-service-sigma.csv
Normal file
File diff suppressed because it is too large
Load Diff
876
docs/operatii-service/operatii-service-south.csv
Normal file
876
docs/operatii-service/operatii-service-south.csv
Normal file
@@ -0,0 +1,876 @@
|
||||
" ";"DENOP";"NR"
|
||||
"51";"DIAGNOZA";"809"
|
||||
"160";"EFECTUAT REVIZIE PERIODICA MICA";"279"
|
||||
"159";"EFECTUAT REVIZIE PERIODICA";"214"
|
||||
"152";"EFECTUARE REVIZIE PERIODICA-MARE";"48"
|
||||
"57";"DIAGNOZA SISTEM AD-BLUE SI SCR";"36"
|
||||
"156";"EFECTUAT REVIZIE MICA";"34"
|
||||
"153";"EFECTUARE REVIZIE PERIODICA-MICA";"33"
|
||||
"625";"INLOCUIT SONDA ADBLUE";"29"
|
||||
"801";"REVIZIE ULEI + FILTRE";"28"
|
||||
"791";"REVIZIE MICA PERIODICA";"28"
|
||||
"792";"REVIZIE MICA PROGRAMATA";"27"
|
||||
"805";"SCHIMB ULEI + FILTRE";"24"
|
||||
"155";"EFECTUAT REVIZIE MARE";"24"
|
||||
"598";"INLOCUIT SENZOR NOX 1";"23"
|
||||
"769";"REPARAT SCARA SPATE STG.";"23"
|
||||
"768";"REPARAT SCARA SPATE DRT.";"23"
|
||||
"149";"DTC CONTROL";"22"
|
||||
"164";"E-PDI";"20"
|
||||
"681";"P.D.I.";"20"
|
||||
"599";"INLOCUIT SENZOR NOX 2";"18"
|
||||
"527";"INLOCUIT POMPA ADBLUE";"18"
|
||||
"789";"REVIZIE MARE PROGRAMATA";"17"
|
||||
"524";"INLOCUIT PLACUTE FRANA FATA";"17"
|
||||
"49";"DEZANSAMBLAT ARC AXA SPATE";"15"
|
||||
"168";"FSA 202322 - ECU UPDATE";"15"
|
||||
"545";"INLOCUIT PTO";"13"
|
||||
"525";"INLOCUIT PLACUTE FRANA SPATE";"13"
|
||||
"171";"FSA 202520-F3V17 BCM UPDATE";"13"
|
||||
"788";"REVIZIE MARE";"11"
|
||||
"827";"VERIFICARE FILTRU ULEI MOTOR";"11"
|
||||
"439";"INLOCUIT KIT AMBREIAJ";"11"
|
||||
"453";"INLOCUIT LAMPA GABARIT CABINA DRT.";"10"
|
||||
"794";"REVIZIE PERIODICA ISUZU";"10"
|
||||
"253";"INLOCUIT BECURI LAMPI SPATE";"10"
|
||||
"705";"REGENERARE FORTATA";"10"
|
||||
"103";"D/R CV DE PE AUTO";"9"
|
||||
"173";"FSA202511A-F3V17 UPDATE BCM";"9"
|
||||
"704";"REGENERARE";"9"
|
||||
"661";"INLOCUIT VALVA RAMPA COMBUSTIBIL";"9"
|
||||
"817";"UPDATE BCM";"9"
|
||||
"435";"INLOCUIT INJECTOR ADBLUE";"9"
|
||||
"798";"REVIZIE PROGRAMATA FILTRE + ULEI";"8"
|
||||
"835";"VERIFICARE PIERDERI ULEI MOTOR";"8"
|
||||
"31";"DEMONTAT ARC SPATE";"8"
|
||||
"151";"EFECTUARE REVIZIE PERIODICA";"8"
|
||||
"454";"INLOCUIT LAMPA GABARIT CABINA STG.";"8"
|
||||
"819";"UPDATE ECM";"7"
|
||||
"515";"INLOCUIT PLACA RELEE MARE";"7"
|
||||
"409";"INLOCUIT FOAIE DE ARC SPATE";"7"
|
||||
"410";"INLOCUIT FOAIE 1 ARC SPATE";"7"
|
||||
"70";"D/R ARC SPATE";"7"
|
||||
"440";"INLOCUIT KIT AMBREIAJ + VOLANTA";"7"
|
||||
"203";"INLOCUIT ALTERNATOR";"7"
|
||||
"441";"INLOCUIT KIT AMBREIAJ, VOLANTA + RULMENT VOLANTA";"7"
|
||||
"799";"REVIZIE PROGRAMATA TRAKKER";"6"
|
||||
"743";"REPARAT CABLAJ LAMPI SPATE";"6"
|
||||
"597";"INLOCUIT SENZOR NOX UPSTREAM DOC 1";"6"
|
||||
"218";"INLOCUIT ARC FATA FORD";"6"
|
||||
"55";"DIAGNOZA RESETARE INTERVAL";"6"
|
||||
"618";"INLOCUIT SIMERING ARBORE SPATE";"6"
|
||||
"832";"VERIFICARE MECANICA GENERALA";"5"
|
||||
"273";"INLOCUIT BUJII MOTOR";"5"
|
||||
"72";"D/R BARA FATA";"5"
|
||||
"75";"D/R BARA PROTECTIE FATA";"5"
|
||||
"543";"INLOCUIT PROIECTOR LUCRU DRT.";"5"
|
||||
"752";"REPARAT CV LA BANC";"5"
|
||||
"455";"INLOCUIT LAMPA LATERALA GABARIT";"5"
|
||||
"186";"INDREPTAT SUPORTI + MONTAT ARIPA DRT. PUNTE MOTRICA";"5"
|
||||
"170";"FSA 202511A - F3V17 BCMUPDATE";"5"
|
||||
"464";"INLOCUIT LAMPA SPATE STG";"5"
|
||||
"800";"REVIZIE PROGRAMATA ULEI + FILTRE";"5"
|
||||
"634";"INLOCUIT SUPAPA RETUR POMPA ADBLUE";"5"
|
||||
"330";"INLOCUIT CONDUCTA A/C";"5"
|
||||
"178";"GOLIT REZERVOR COMBUSTIBIL";"4"
|
||||
"862";"VERIFICAT SI INCARCAT INSTALATIE CU A/C";"4"
|
||||
"100";"D/R CUTIE DE VITEZE";"4"
|
||||
"234";"INLOCUIT BASCULA DR FATA";"4"
|
||||
"169";"FSA 202413";"4"
|
||||
"588";"INLOCUIT SENZOR ABS SPATE DR";"4"
|
||||
"611";"INLOCUIT SENZORI PLACUTE FATA";"4"
|
||||
"462";"INLOCUIT LAMPA SPATE DR";"4"
|
||||
"839";"VERIFICARE SI INCARCARE INSTALATIE A/C";"4"
|
||||
"375";"INLOCUIT DISCURI FRINA SPATE";"4"
|
||||
"607";"INLOCUIT SENZOR TEMPERATURA CATALIZATOR";"4"
|
||||
"829";"VERIFICARE INSTALATIE A/C + INCARCARE CU FREON";"4"
|
||||
"828";"VERIFICARE GENERALA";"4"
|
||||
"552";"INLOCUIT RADIATOR RACIRE MOTOR";"4"
|
||||
"25";"CURATAT REZERVOR ADBLUE";"4"
|
||||
"402";"INLOCUIT FILTRU EPURATOR";"4"
|
||||
"857";"VERIFICAT INSTALATIE AC + INCARCAT FREON";"4"
|
||||
"654";"INLOCUIT TURBOSUFLANTA";"4"
|
||||
"683";"PIAGGIO SCHIMB ULEI + FILTRE";"4"
|
||||
"821";"UPDATE TAQ 766187";"4"
|
||||
"78";"D/R BORD AUTO";"4"
|
||||
"530";"INLOCUIT POMPA APA";"4"
|
||||
"813";"TELESERVICII PENTRU LIMITARE";"4"
|
||||
"617";"INLOCUIT SIMERING ARBORE FATA";"4"
|
||||
"824";"VERIF SI INCARCAT AC";"4"
|
||||
"71";"D/R ARIPA FATA STG";"4"
|
||||
"595";"INLOCUIT SENZOR NOX DUPA DPF (ULTIMUL)";"3"
|
||||
"495";"INLOCUIT OGLINDA PIETON";"3"
|
||||
"139";"D/R ROTI AXA 2FATA STG/DR";"3"
|
||||
"521";"INLOCUIT PLACUTE FRANA";"3"
|
||||
"756";"REPARAT INSTALATIE ELECTRICA LAMPI SPATE";"3"
|
||||
"501";"INLOCUIT ORNAMENT ARIPA FATA";"3"
|
||||
"468";"INLOCUIT LEVIER COMANDA + CAPETE DE BARA TRANSV";"3"
|
||||
"549";"INLOCUIT RADIATOR AEROTERMA";"3"
|
||||
"467";"INLOCUIT LAMPI SPATE";"3"
|
||||
"596";"INLOCUIT SENZOR NOX INAINTE DE DPF";"3"
|
||||
"179";"GRESAT";"3"
|
||||
"489";"INLOCUIT OCHELAR PROIECTOR DRT.";"3"
|
||||
"478";"INLOCUIT MANETA SCHIMBATOR VITEZE";"3"
|
||||
"484";"INLOCUIT MONITOR + CAMERA";"3"
|
||||
"673";"MATERIALE ELECTRICE";"3"
|
||||
"290";"INLOCUIT CAMERA SPATE";"3"
|
||||
"575";"INLOCUIT RULMENTI INTERMEDIARI CARDAN";"3"
|
||||
"573";"INLOCUIT RULMENTI AXA SPATE";"3"
|
||||
"698";"REFACUT CABLAJ ELECTRIC USA STG";"3"
|
||||
"675";"MONTAT ARC AXA SPATE";"3"
|
||||
"581";"INLOCUIT SABOTI FRANA MANA";"3"
|
||||
"331";"INLOCUIT CONDUCTA AC";"3"
|
||||
"343";"INLOCUIT CONTACT PEDALA FRANA";"3"
|
||||
"670";"MANOPERA";"3"
|
||||
"666";"INLOCUIT VOLANTA MOTOR";"3"
|
||||
"662";"INLOCUIT VAS EXPANSIUNE";"3"
|
||||
"741";"REPARAT CABLAJ ELECTRIC LA SENZORUL DE PLACUTE STG FATA";"3"
|
||||
"610";"INLOCUIT SENZORI NOX 1 SI 2";"3"
|
||||
"394";"INLOCUIT FILTRU ADBLUE";"3"
|
||||
"184";"INCARCAT INST A/C + VERIFICAT";"3"
|
||||
"187";"INDREPTAT SUPORTI + MONTAT ARIPA STG. PUNTE MOTRICA";"3"
|
||||
"373";"INLOCUIT DISCURI FRANA STG + DR SPATE";"3"
|
||||
"715";"REMEDIAT CABLAJ PANOU COMANDA SUPRASTRUCTURA";"3"
|
||||
"710";"REGLAT FARURUI";"3"
|
||||
"249";"INLOCUIT BECURI";"3"
|
||||
"222";"INLOCUIT ARIPA + ORNAMENT DRT. PUNTE FATA";"3"
|
||||
"233";"INLOCUIT BARA FATA STG";"3"
|
||||
"235";"INLOCUIT BASCULA INFERIOARA FATA STG";"3"
|
||||
"132";"D/R REZERVOR COMBUSTIBIL";"3"
|
||||
"131";"D/R REZERVOR ADBLUE";"3"
|
||||
"779";"REPARATII ELECTRICE";"3"
|
||||
"822";"UPDATE TAS 766161";"3"
|
||||
"101";"D/R CUTIE VITEZA SI INLOCUIT BOLT SELECTOR";"3"
|
||||
"4";"AERISIT INSTALATIE RACIRE";"3"
|
||||
"122";"D/R MOTOR SI ACCESORII MOTOR";"3"
|
||||
"87";"D/R CARDAN";"3"
|
||||
"806";"SERVICE SCHIMB ULEI";"3"
|
||||
"107";"D/R FATA DE USA STANGA";"3"
|
||||
"16";"CURATAT INSTALATIE ALIMENTARE COMBUSTIBIL";"3"
|
||||
"796";"REVIZIE PERIODICA OTOKAR";"3"
|
||||
"288";"INLOCUIT CABLURI TIMONERIE";"2"
|
||||
"560";"INLOCUIT ROLA AC+CUREA";"2"
|
||||
"264";"INLOCUIT BUCSI BARA TORSIUINE FATA STG";"2"
|
||||
"443";"INLOCUIT KIT CUREA DISTRIBUTIE";"2"
|
||||
"699";"REFACUT CABLAJ LAMPA SPATE STG";"2"
|
||||
"671";"MANOPERA ADITIONALA";"2"
|
||||
"424";"INLOCUIT GARNITURI DPF";"2"
|
||||
"433";"INLOCUIT IMPULSOR CV";"2"
|
||||
"83";"D/R CAPAC OGLINDA INF STG";"2"
|
||||
"569";"INLOCUIT RULMENT ROATA FATA STG";"2"
|
||||
"267";"INLOCUIT BUCSI CABINA FATA";"2"
|
||||
"562";"INLOCUIT ROLE + CUREA";"2"
|
||||
"434";"INLOCUIT INJECTOARE";"2"
|
||||
"81";"D/R CADRU FATA AUTO";"2"
|
||||
"456";"INLOCUIT LAMPA NR DR";"2"
|
||||
"457";"INLOCUIT LAMPA NR STG";"2"
|
||||
"714";"REMEDIAT CABLAJ ELECTRIC LAMPI SPATE";"2"
|
||||
"859";"VERIFICAT INSTALATIE COMPACTARE";"2"
|
||||
"713";"REMEDIAT CABLAJ ELECTRIC";"2"
|
||||
"863";"VERIFICAT SI REPARAT INSTALATIE GIDRAULICA SUPRASTRUCTURA";"2"
|
||||
"723";"REMEDIAT PIERDERI AER PE INSTALATIA DE FRANARE";"2"
|
||||
"861";"VERIFICAT INSTALATIE ELECTRICA SISTEM EVACUARE";"2"
|
||||
"803";"REVIZIE 40 000KM";"2"
|
||||
"550";"INLOCUIT RADIATOR APA";"2"
|
||||
"858";"VERIFICAT INSTALATIE ALIMENTARE CU AD-BLUE";"2"
|
||||
"447";"INLOCUIT KIT SINCROANE REDUCTOR CV";"2"
|
||||
"448";"INLOCUIT KIT SINCROANE VIT. 1- 2";"2"
|
||||
"558";"INLOCUIT REZERVOR COMBUSTIBIL";"2"
|
||||
"255";"INLOCUIT BOLTURI ETRIER SPATE";"2"
|
||||
"444";"INLOCUIT KIT DISTRIBUTIE + POMPA DE APA";"2"
|
||||
"856";"VERIFICAT INSTALATIE A/C";"2"
|
||||
"808";"SERVICII VOPSITORIE";"2"
|
||||
"554";"INLOCUIT REGULATOR PRESIUNE AER";"2"
|
||||
"90";"D/R CHIULASA + GARNITURA";"2"
|
||||
"450";"INLOCUIT LAMPA CABINA STG";"2"
|
||||
"291";"INLOCUIT CAMERA VIDEO";"2"
|
||||
"345";"INLOCUIT CRABOTI SELECTOR CV";"2"
|
||||
"344";"INLOCUIT CONTACTOR PEDALA FRANA";"2"
|
||||
"346";"INLOCUIT CRUCE CARDAN";"2"
|
||||
"627";"INLOCUIT SONDA LITROMETRICA";"2"
|
||||
"605";"INLOCUIT SENZOR PTO";"2"
|
||||
"338";"INLOCUIT CONDUCTA RACIRE Y";"2"
|
||||
"600";"INLOCUIT SENZOR PLACUTE";"2"
|
||||
"601";"INLOCUIT SENZOR PM";"2"
|
||||
"342";"INLOCUIT CONDUCTE RACIRE";"2"
|
||||
"602";"INLOCUIT SENZOR PRESIUNE";"2"
|
||||
"358";"INLOCUIT CUZINETI ARBORE COTIT";"2"
|
||||
"372";"INLOCUIT DISCURI FRANA FATA";"2"
|
||||
"360";"INLOCUIT CUZINETI BIELE";"2"
|
||||
"368";"INLOCUIT DISCURI + ROTI AXA SPATE";"2"
|
||||
"369";"INLOCUIT DISCURI + SET PLACUTE FRANA SPATE";"2"
|
||||
"44";"DEMONTAT/MONTAT CV";"2"
|
||||
"58";"DIAGNOZA SISTEM SCR";"2"
|
||||
"623";"INLOCUIT SIMERING SI INEL RT/SP/DR";"2"
|
||||
"374";"INLOCUIT DISCURI FRINA FATA";"2"
|
||||
"376";"INLOCUIT DISCURI SI PLACUTE FRANA FATA";"2"
|
||||
"633";"INLOCUIT SUPAPA MODUL ADBLUE";"2"
|
||||
"303";"INLOCUIT CAPETE TIRANTI";"2"
|
||||
"664";"INLOCUIT VASCOCUPLAJ VENTILATOR";"2"
|
||||
"658";"INLOCUIT ULEI MOTOR + FILTRU";"2"
|
||||
"651";"INLOCUIT TERMOSTATE SISTEM DE RACIRE";"2"
|
||||
"407";"INLOCUIT FILTRU ULEI HIDRAULIC";"2"
|
||||
"294";"INLOCUIT CAP BARA DR";"2"
|
||||
"77";"D/R BARA TORSIUNE STG";"2"
|
||||
"295";"INLOCUIT CAP BARA STG";"2"
|
||||
"665";"INLOCUIT VENTILATOR AEROTERMA";"2"
|
||||
"76";"D/R BARA TORSIUNE DR";"2"
|
||||
"594";"INLOCUIT SENZOR NIVEL LICHID RACIRE OTOKAR";"2"
|
||||
"325";"INLOCUIT COMPRESOR AC";"2"
|
||||
"390";"INLOCUIT FAR FAZA SCURTA STG.";"2"
|
||||
"332";"INLOCUIT CONDUCTA CU SENZOR INCALZIRE ADBLUE";"2"
|
||||
"39";"DEMONTAT SCAUN SOFER SI INLOCUIT PERNA SEZUT";"2"
|
||||
"399";"INLOCUIT FILTRU DE PARTICULE";"2"
|
||||
"403";"INLOCUIT FILTRU HIDRAULIC";"2"
|
||||
"397";"INLOCUIT FILTRU COMBUSTIBIL";"2"
|
||||
"591";"INLOCUIT SENZOR FILTRU COMBUSTIBIL + FILTRU";"2"
|
||||
"396";"INLOCUIT FILTRU AER";"2"
|
||||
"158";"EFECTUAT REVIZIE M2";"2"
|
||||
"194";"INLOCUIRE GARNITURA CULBUTORI";"2"
|
||||
"748";"REPARAT CABLAJ SUPRASTRUCTURA";"2"
|
||||
"127";"D/R PRAG USA DR";"2"
|
||||
"165";"FACTURA TRACTARE ANB0093/07.04.2026";"2"
|
||||
"502";"INLOCUIT ORNAMENT ARIPA STG. PUNTE FATA";"2"
|
||||
"534";"INLOCUIT POMPA DE ULEI";"2"
|
||||
"157";"EFECTUAT REVIZIE MOTOR AUXILIAR";"2"
|
||||
"129";"D/R RADIATOARE + CONDENSOARE";"2"
|
||||
"507";"INLOCUIT PINION VIT 1";"2"
|
||||
"104";"D/R DIFERENTIAL AXA SPATE";"2"
|
||||
"536";"INLOCUIT POMPA ULEI";"2"
|
||||
"868";"VF. + INLOCUIT MONITOR";"2"
|
||||
"746";"REPARAT CABLAJ SENZORI PLACUTE FRANA";"2"
|
||||
"745";"REPARAT CABLAJ MOTOR";"2"
|
||||
"167";"FSA 202415B1";"2"
|
||||
"172";"FSA202420-INLOCUIT FLANSA + BOLT CARDAN";"2"
|
||||
"8";"COMPLETARE FREON + VERIFICARE INST A/C";"2"
|
||||
"786";"REVIZIE EO-MPE00000";"2"
|
||||
"870";"VF. SI INCARCAT SISTEM AC CU FREON";"2"
|
||||
"176";"GOLIT INSTALATIE RACIRE MOT.";"2"
|
||||
"174";"GOLIT INSTALATIE A/C";"2"
|
||||
"121";"D/R MOTOR DE PE AUTO";"2"
|
||||
"757";"REPARAT INSTALATIE ELECTRICA SUPRASTRUCTURA";"2"
|
||||
"784";"REVIZIE ULEI + FILTRE";"2"
|
||||
"797";"REVIZIE PERIODICA ULEI SI FILTRE";"2"
|
||||
"869";"VF. + REMEDIAT I.E. ILUMINAT EXTERIOR";"2"
|
||||
"490";"INLOCUIT OCHELAR PROIECTOR STG.";"2"
|
||||
"492";"INLOCUIT OGLINDA BORDURA";"2"
|
||||
"499";"INLOCUIT OGLINDA STG ELECTRICA";"2"
|
||||
"531";"INLOCUIT POMPA APA MOTOR";"2"
|
||||
"224";"INLOCUIT ARIPA FATA STG";"2"
|
||||
"219";"INLOCUIT ARC SPATE";"2"
|
||||
"475";"INLOCUIT MACARA STG";"2"
|
||||
"512";"INLOCUIT PIVOTI INF + SUP (AMBELE PARTI)";"2"
|
||||
"466";"INLOCUIT LAMPI LATERALE GABARIT";"2"
|
||||
"514";"INLOCUIT PLACA RELEE";"2"
|
||||
"14";"CURATAT INSTALATIA ADMISIE";"2"
|
||||
"138";"D/R ROTI AXA FATA STG/DR";"2"
|
||||
"469";"INLOCUIT LEVIER DE COMANDA";"2"
|
||||
"98";"D/R CONDUCTE COMBUSTIBIL PARTIAL";"2"
|
||||
"771";"REPARAT TOBA FINALA";"2"
|
||||
"519";"INLOCUIT PLACUTE FR FATA";"2"
|
||||
"133";"D/R RIGIDIZARE TREAPTA STG";"2"
|
||||
"520";"INLOCUIT PLACUTE FR FATA + SPATE";"2"
|
||||
"729";"REMEDIAT PIERDERI ULEI CILINDRU COMPACTARE STG.";"2"
|
||||
"776";"REPARATIE GRUP DIFERENTIAL";"2"
|
||||
"510";"INLOCUIT PIVOTI AXA FATA";"2"
|
||||
"511";"INLOCUIT PIVOTI FATA STG + DR";"2"
|
||||
"99";"D/R CUTIE DE VITEZA AUTO";"2"
|
||||
"217";"INLOCUIT ARBORE MOTOR";"2"
|
||||
"206";"INLOCUIT AMORTIZOARE AXA FATA";"2"
|
||||
"609";"INLOCUIT SENZOR ULEI MOTOR";"1"
|
||||
"517";"INLOCUIT PLACUTE AXA FATA";"1"
|
||||
"586";"INLOCUIT SENZOR ABS DR SPATE";"1"
|
||||
"587";"INLOCUIT SENZOR ABS FATA DR";"1"
|
||||
"834";"VERIFICARE PIERDERI ULEI";"1"
|
||||
"526";"INLOCUIT PLACUTE FRINA";"1"
|
||||
"846";"VERIFICARI ELECTRICE SI UDT";"1"
|
||||
"836";"VERIFICARE PRESIUNE INSTALATIE ADBLUE";"1"
|
||||
"590";"INLOCUIT SENZOR AXA CAME";"1"
|
||||
"845";"VERIFICARE ULEI GRUP DIFERENTIAL";"1"
|
||||
"589";"INLOCUIT SENZOR AMONIAC";"1"
|
||||
"518";"INLOCUIT PLACUTE AXA SPATE";"1"
|
||||
"522";"INLOCUIT PLACUTE FRANA + AERISIT";"1"
|
||||
"523";"INLOCUIT PLACUTE FRANA AXA FATA";"1"
|
||||
"841";"VERIFICARE SISTEM ALIMENTARE";"1"
|
||||
"840";"VERIFICARE SISTEM A/C";"1"
|
||||
"604";"INLOCUIT SENZOR PRESIUNE ULEI";"1"
|
||||
"874";"VOPSIT PRIZA AER STG";"1"
|
||||
"606";"INLOCUIT SENZOR STANGA FATA";"1"
|
||||
"838";"VERIFICARE SERVODIRECTIE";"1"
|
||||
"603";"INLOCUIT SENZOR PRESIUNE DIFERENTIALA";"1"
|
||||
"593";"INLOCUIT SENZOR LIFT PUBELE";"1"
|
||||
"872";"VOPSIT BARA FATA STG";"1"
|
||||
"592";"INLOCUIT SENZOR LICHID DE RACIRE";"1"
|
||||
"844";"VERIFICARE ULEI CV";"1"
|
||||
"608";"INLOCUIT SENZOR ULEI";"1"
|
||||
"837";"VERIFICARE SENZORI UZURA SI ABS";"1"
|
||||
"842";"VERIFICARE SISTEM ELECTRIC AD-BLUE";"1"
|
||||
"873";"VOPSIT CAPOTA MOTOR";"1"
|
||||
"875";"VOPSITORIE + MATERIALE VOPSITORIE";"1"
|
||||
"843";"VERIFICARE SISTEM FRANARE";"1"
|
||||
"585";"INLOCUIT SENZOR + CABLAJ NOX";"1"
|
||||
"539";"INLOCUIT PRESOSTAT";"1"
|
||||
"538";"INLOCUIT POMPITA STERGATOR";"1"
|
||||
"537";"INLOCUIT POMPITA SPALATOR PARBRIZ";"1"
|
||||
"867";"VERIFICAT UZURA CILINDRI SI BLOC MOTOR";"1"
|
||||
"557";"INLOCUIT RELEU BUJII";"1"
|
||||
"540";"INLOCUIT PREZOANE ROATA DR";"1"
|
||||
"559";"INLOCUIT REZISTENTA AEROTERMA";"1"
|
||||
"561";"INLOCUIT ROLA GHIDARE + CUREA TRANSMISIE";"1"
|
||||
"852";"VERIFICAT INCARCARE ALTERNATOR";"1"
|
||||
"563";"INLOCUIT RULMENT AMBREIAJ";"1"
|
||||
"855";"VERIFICAT INSTALATI ELECTRICA A/C";"1"
|
||||
"854";"VERIFICAT INJECTOARE";"1"
|
||||
"853";"VERIFICAT/ INCARCAT INSTALATIE A/C";"1"
|
||||
"541";"INLOCUIT PROIECTOARE";"1"
|
||||
"860";"VERIFICAT INSTALATIE ELECTRICA SI INLOCUIT BUTON AVARIE";"1"
|
||||
"553";"INLOCUIT RAMA PROIECTOR + BEC";"1"
|
||||
"547";"INLOCUIT RACORD RACIRE EGR";"1"
|
||||
"548";"INLOCUIT RADIATOR A/C";"1"
|
||||
"864";"VERIFICAT SISTEM ALIMENTARE";"1"
|
||||
"551";"INLOCUIT RADIATOR RACIRE";"1"
|
||||
"546";"INLOCUIT RACORD FLEXIBIL EVACUARE";"1"
|
||||
"555";"INLOCUIT REGULATOR PRESIUNE COMBUSTIBIL";"1"
|
||||
"556";"INLOCUIT RELEE - 2 BUC";"1"
|
||||
"542";"INLOCUIT PROIECTOR CEATA STG.";"1"
|
||||
"865";"VERIFICAT SISTEM EVACUARE";"1"
|
||||
"544";"INLOCUIT PROTECTII FOAIE DE ARC TRANSVERSALA";"1"
|
||||
"866";"VERIFICAT SISTEM FRANARE";"1"
|
||||
"564";"INLOCUIT RULMENT AXA SPATE";"1"
|
||||
"580";"INLOCUIT SABOTI AXA SPATE";"1"
|
||||
"529";"INLOCUIT POMPA AMOR";"1"
|
||||
"849";"VERIFICAT BUJII + APRINDERE";"1"
|
||||
"577";"INLOCUIT RULMENTI ROTI SPATE";"1"
|
||||
"578";"INLOCUIT RULMRNT PRESIUNE AMBREIAJ";"1"
|
||||
"579";"INLOCUIT RULMRNT PRIZA CV";"1"
|
||||
"582";"INLOCUIT SABOTI SPATE";"1"
|
||||
"847";"VERIFICAT + AERISIT SIST FRANARE";"1"
|
||||
"871";"VF. SI REMEDIAT CABLAJ ELECTROMOTOR";"1"
|
||||
"584";"INLOCUIT SEMNALIZARE DR PE OGLINDA";"1"
|
||||
"528";"INLOCUIT POMPA AMBREIAJ";"1"
|
||||
"583";"INLOCUIT SELECTOR VITEZE";"1"
|
||||
"848";"VERIFICAT ADMISIE AER";"1"
|
||||
"576";"INLOCUIT RULMENTI ROTI FATA AMBELE PARTI";"1"
|
||||
"568";"INLOCUIT RULMENT ROATA FATA DR";"1"
|
||||
"535";"INLOCUIT POMPA INALTA PRESIUNE";"1"
|
||||
"570";"INLOCUIT RULMENT ROATA SPATE";"1"
|
||||
"565";"INLOCUIT RULMENT BUTUC DR. SPATE";"1"
|
||||
"566";"INLOCUIT RULMENT DIFERENTIAL";"1"
|
||||
"567";"INLOCUIT RULMENT GRUP DIF.";"1"
|
||||
"533";"INLOCUIT POMPA DE AMORSARE INST DE COMBUSTIBIL";"1"
|
||||
"574";"INLOCUIT RULMENTI FATA (AMBELE PARTI)";"1"
|
||||
"850";"VERIFICAT CABLAJ ELECTRIC";"1"
|
||||
"532";"INLOCUIT POMPA DE AMBREIAJ";"1"
|
||||
"571";"INLOCUIT RULMENT VOLANTA";"1"
|
||||
"572";"INLOCUIT RULMENTI AXA FATA";"1"
|
||||
"851";"VERIFICAT CONCENTRATIE + COMPLETAT ANTIGEL";"1"
|
||||
"726";"REMEDIAT PIERDERI DE AER PE SISTEMUL DE FRANARE";"1"
|
||||
"725";"REMEDIAT PIERDERI APA LA INSTALATIA DE SPALAT";"1"
|
||||
"728";"REMEDIAT PIERDERI ULEI CILINDRU ACTIONARE LIFT STG.";"1"
|
||||
"727";"REMEDIAT PIERDERI DE ULEI MOTOR";"1"
|
||||
"724";"REMEDIAT PIERDERI ANTIGEL";"1"
|
||||
"720";"REMEDIAT INSTALATIE ELECTRICA";"1"
|
||||
"719";"REMEDIAT FCT. PROIECTOARE SPATE";"1"
|
||||
"722";"REMEDIAT LUMINI MI";"1"
|
||||
"721";"REMEDIAT INTRERUPERI CABLAJ MOTOR";"1"
|
||||
"736";"REPARAT ARIPA SPATE STG";"1"
|
||||
"735";"REPARAT ARC DR FATA";"1"
|
||||
"738";"REPARAT BARA STG. FATA";"1"
|
||||
"737";"REPARAT BARA DRT. FATA";"1"
|
||||
"734";"REMEDIERE PRINDERE COMP AC";"1"
|
||||
"731";"REMEDIAT PRINDERE COLIER TURBOSUFLANTA";"1"
|
||||
"730";"REMEDIAT PIULITA GRUP DIFERENTIAL";"1"
|
||||
"733";"REMEDIAT PRINDERI CONDUCTA A/C";"1"
|
||||
"732";"REMEDIAT PRINDERI ACTIONARE LIFT SUPERIOARE STG. + DRT.";"1"
|
||||
"718";"REMEDIAT DEFECTIUNI PORNIRE";"1"
|
||||
"810";"SPALAT MOTOR CU SOLUTIE";"1"
|
||||
"703";"REFACUT SUPORT TABLOU SIGURANTE";"1"
|
||||
"706";"REGLARE FARURI";"1"
|
||||
"809";"SPALAT INSTALATIE ALIMENTARE AD-BLUE";"1"
|
||||
"702";"REFACUT INSTALATIE ELECTRICA SUPRASTRUCTURA";"1"
|
||||
"700";"REFACUT CABLAJ SENZOR PLACUTE SPATE";"1"
|
||||
"697";"REFACTURARE CURATARE DPF";"1"
|
||||
"811";"TELESERVICII";"1"
|
||||
"701";"REFACUT CABLAJ TABLOU CMD. HYDRO-MAK";"1"
|
||||
"716";"REMEDIAT CABLAJ SENZOR MARSARIER";"1"
|
||||
"807";"SERVICII TERTI FACTURA 30/14.02.2025 REPARATIE INJECTOARE";"1"
|
||||
"804";"REVOPSIT USA STG FATA";"1"
|
||||
"717";"REMEDIAT CONDUCTA EVACURE";"1"
|
||||
"712";"REGLAT VOLAN";"1"
|
||||
"708";"REGLAT CULBUTORI";"1"
|
||||
"707";"REGLAT CABLURI FRANA";"1"
|
||||
"711";"REGLAT FRANA MANA";"1"
|
||||
"709";"REGLAT FARURI";"1"
|
||||
"739";"REPARAT CABLAJ LAMPA SPATE";"1"
|
||||
"782";"RESOFTARE";"1"
|
||||
"765";"REPARAT PARABICICLISTI DR";"1"
|
||||
"766";"REPARAT PRINDERE GIROFAR FATA";"1"
|
||||
"781";"RESETAT INTERVAL REVIZIE";"1"
|
||||
"764";"REPARAT PANOU COMANDA COMPACTOR";"1"
|
||||
"762";"REPARAT MUFE ELECTRICE BOBINE";"1"
|
||||
"785";"REVIZIE DIFERENTIAL";"1"
|
||||
"763";"REPARAT ORNAMENT TOBA";"1"
|
||||
"783";"REV 20000";"1"
|
||||
"774";"REPARATIE BOLT SUPRASTRUCTURA COMPACTARE";"1"
|
||||
"773";"REPARAT USA STG FATA";"1"
|
||||
"777";"REPARATIE LONJERON STG";"1"
|
||||
"775";"REPARATIE CABLAJ ELECTRIC";"1"
|
||||
"772";"REPARAT TRAVERSA SPATE SASIU";"1"
|
||||
"780";"REPROGRAMARE BODY COMPUTER";"1"
|
||||
"767";"REPARAT SCARA DREAPTA";"1"
|
||||
"770";"REPARAT SISTEM EVACUARE";"1"
|
||||
"778";"REPARATIE LUMINI LAMPI SPATE";"1"
|
||||
"761";"REPARAT MUFA ELECTRICA PE CABLAJ NOX";"1"
|
||||
"750";"REPARAT CONDUCTA SISTEM PNEUMATIC SPATE";"1"
|
||||
"749";"REPARAT CABLAJ TABLOU CMD.";"1"
|
||||
"753";"REPARAT INST. ELECTRICA SENZOR PEDALA FR";"1"
|
||||
"751";"REPARAT CUTIE VITEZE";"1"
|
||||
"747";"REPARAT CABLAJ SI INLOCUIT LAMPA SPATE";"1"
|
||||
"802";"REVIZIE ULEI +FILTRE";"1"
|
||||
"740";"REPARAT CABLAJ COMUTATOR FTANA MANA";"1"
|
||||
"744";"REPARAT CABLAJ LUMINI SPATE";"1"
|
||||
"742";"REPARAT CABLAJ ELECTRIC SISTEM EVACUARE";"1"
|
||||
"790";"REVIZIE MICA - PROGRAMATA";"1"
|
||||
"759";"REPARAT LIFT SUPRASTRUCTURA";"1"
|
||||
"760";"REPARAT MANETA COMENZI HIDRAULICE";"1"
|
||||
"787";"REVIZIE FILTRE + ULEI";"1"
|
||||
"758";"REPARAT INSTALATIE HIDRAULICA SUPRASTRUCTURA";"1"
|
||||
"755";"REPARAT INSTALATIE ELECTRICA";"1"
|
||||
"754";"REPARAT INSTALATIE ELECRICA ILUMINAT";"1"
|
||||
"793";"REVIZIE MOTOR AUXILIAR";"1"
|
||||
"795";"REVIZIE PERIODICA MARE";"1"
|
||||
"696";"REFACTURARE ALEZAJ BLOC MOTOR";"1"
|
||||
"638";"INLOCUIT SUPORT FAR STG";"1"
|
||||
"637";"INLOCUIT SUPORT CAPOTA INFERIOR STG";"1"
|
||||
"640";"INLOCUIT SUPORTI BARA SPATE STG + DR";"1"
|
||||
"639";"INLOCUIT SUPORT INF. DR CAPOTA";"1"
|
||||
"823";"UPDATE VBR 768840";"1"
|
||||
"635";"INLOCUIT SUPAPA SISTEM ALIMENTARE RAMPA";"1"
|
||||
"826";"VERIFICARE ELECTRICA";"1"
|
||||
"825";"VERIFICARE AUTO/CONSTATARE";"1"
|
||||
"636";"INLOCUIT SUPAPA SUPRASTRUCTURA";"1"
|
||||
"647";"INLOCUIT TAMPOANE CABINA FATA";"1"
|
||||
"646";"INLOCUIT TAMPOANE ARCURI SPATE";"1"
|
||||
"649";"INLOCUIT TAMPOANE SUPERIOARE RIGIDIZARE FATA";"1"
|
||||
"648";"INLOCUIT TAMPOANE INFERIOARE PUNTE FATA";"1"
|
||||
"645";"INLOCUIT TAMBURI SPATE AXA 3";"1"
|
||||
"642";"INLOCUIT SURUBURI FUZETA SPATE";"1"
|
||||
"641";"INLOCUIT SURUB AMORTIZOR FATA STG/DR";"1"
|
||||
"644";"INLOCUIT TAMBURI FRANA AXA 2 FATA";"1"
|
||||
"643";"INLOCUIT SURUBURI VOLANTA";"1"
|
||||
"632";"INLOCUIT SUPAPA GV";"1"
|
||||
"831";"VERIFICARE INSTALATIE ELECTRA SISTEM AD-BLUE";"1"
|
||||
"833";"VERIFICARE PIERDERI COMBUSTIBIL";"1"
|
||||
"620";"INLOCUIT SIMERING BUTUC SPATE";"1"
|
||||
"619";"INLOCUIT SIMERING AX CAME";"1"
|
||||
"616";"INLOCUIT SIMERERING GRUP DIFERENTIAL";"1"
|
||||
"613";"INLOCUIT SENZOT PTO";"1"
|
||||
"612";"INLOCUIT SENZORI TEMPERATURA EVACUARE";"1"
|
||||
"615";"INLOCUIT SIGURANTA 7.5 AH";"1"
|
||||
"614";"INLOCUIT SERPENTINA ADBLUE";"1"
|
||||
"629";"INLOCUIT STUT";"1"
|
||||
"830";"VERIFICARE INSTALATIE ALIMENTARE CU AD-BLUE";"1"
|
||||
"631";"INLOCUIT SUPAPA EGR";"1"
|
||||
"630";"INLOCUIT SUPAPA CUTIE DE VITEZA";"1"
|
||||
"628";"INLOCUIT SONDA REZERVOR";"1"
|
||||
"622";"INLOCUIT SIMERING PALIER SPATE";"1"
|
||||
"621";"INLOCUIT SIMERING FATA MOTOR";"1"
|
||||
"626";"INLOCUIT SONDA LAMBDA";"1"
|
||||
"624";"INLOCUIT SINE SCAUN SOFER";"1"
|
||||
"650";"INLOCUIT TERMOFLOT";"1"
|
||||
"685";"PREGATIRE ELEMENTE DIN PLASTIC";"1"
|
||||
"684";"PIESE MARUNTE";"1"
|
||||
"686";"PREGATIRE ELEMENTE METALICE";"1"
|
||||
"812";"TELESERVICII IVECO INLOCUIRE INSTRUMENTE BORD";"1"
|
||||
"814";"TELESERVICII- SOFT ALM";"1"
|
||||
"679";"MONTAT PARABICICLISTI STG";"1"
|
||||
"678";"MONTAT CONDUCTA ALIMENTARE PENTRU INCALZIRE AUXILIARA";"1"
|
||||
"682";"PIAGGIO INLOCUIT DISCURI SI PLACUTE FRANA FATA";"1"
|
||||
"680";"MONTAT PROIECTOARE FATA";"1"
|
||||
"693";"RAMPA INJECTOARE DEMONTATA";"1"
|
||||
"692";"PURJAT FILTRU MOTORINA";"1"
|
||||
"695";"RECTIFICAT FILETE COMPRESOR A/C";"1"
|
||||
"694";"RECTIFICAT FILET AXA DR SPATE";"1"
|
||||
"691";"PROGRAMARE CHEIE";"1"
|
||||
"688";"PRINDERE CONDUCTA EVACUARE PE SASIU";"1"
|
||||
"687";"PREGATIRE PENTRU VOPSIRE";"1"
|
||||
"690";"PROGRAMARE BODY COMPUTER";"1"
|
||||
"689";"PRINDERE CONDUCTE A/C";"1"
|
||||
"677";"MONTAT BUTUC ROATA SPATE DR";"1"
|
||||
"660";"INLOCUIT USA STG FATA";"1"
|
||||
"659";"INLOCUIT UNITATE CONTROL BUJII";"1"
|
||||
"818";"UPDATE BCM 76618 ATAQ";"1"
|
||||
"820";"UPDATE SOFT FSA 202418";"1"
|
||||
"657";"INLOCUIT ULEI CV";"1"
|
||||
"653";"INLOCUIT TURBINA MOTOR AUXILIAR";"1"
|
||||
"652";"INLOCUIT TREAPTA CABINA ARIPA DR";"1"
|
||||
"656";"INLOCUIT ULEI CUTIE VITEZE";"1"
|
||||
"655";"INLOCUIT ULEI + FILTRU ULEI";"1"
|
||||
"674";"MONTAT APARATOARE NOROI SPATE";"1"
|
||||
"672";"MASURAT UZURA ARBORE SI INLOCUIT CUZINETI PALIER (6 +1 AXIAL)";"1"
|
||||
"676";"MONTAT ARC AXA SPATE DR";"1"
|
||||
"815";"TEST ACUMULATORI";"1"
|
||||
"816";"TEST FRANARE";"1"
|
||||
"667";"INLOCUIT 2 BIELE MOTOR";"1"
|
||||
"663";"INLOCUIT VAS EXPANSIUNE CU SENZOR";"1"
|
||||
"669";"LIMITARE VITEZA LA 110KM/H";"1"
|
||||
"668";"LIMITARE AUTO 110 KM/H";"1"
|
||||
"183";"INCARCAT INSTALATIE A/C CU FREON";"1"
|
||||
"182";"INCARCARE INSTALATIE A/C";"1"
|
||||
"185";"INDREPTAT CAPAC SPATE PARTEA DRT.";"1"
|
||||
"189";"INL. INCHIZ CAPOTA DR";"1"
|
||||
"188";"INL DUBLURA INTERIOARA LONGERON FATA STG";"1"
|
||||
"181";"IINLOCUIT TERMOSTAT";"1"
|
||||
"166";"FACTURA VOPSITORIE F1897";"1"
|
||||
"163";"EFECTUAT TESTE + REGENERARE";"1"
|
||||
"175";"GOLIT INSTALATIE RACIRE";"1"
|
||||
"180";"GRESAT PUNCTE";"1"
|
||||
"177";"GOLIT REZERVOR AD-BLUE";"1"
|
||||
"198";"INLOCUIRE VENTILATOR AEROTERMA";"1"
|
||||
"197";"INLOCUIRE RIGIDIZARE STALP EXT. INT. DR";"1"
|
||||
"199";"INLOCUIT + REGLAT CABLURI TIMONERIE";"1"
|
||||
"201";"INLOCUIT ACUMULATORI";"1"
|
||||
"200";"INLOCUIT ACUMULATOR";"1"
|
||||
"196";"INLOCUIRE PERNA AER PUNTE MOTRICA DRT. SPRE FATA";"1"
|
||||
"191";"INLOCOUIT FAR FAZA LUNGA STG.";"1"
|
||||
"190";"INL. INCHIZ CAPOTA STG";"1"
|
||||
"192";"INLOCUIRE BARA FATA STG";"1"
|
||||
"195";"INLOCUIRE PERNA AER PUNTE MOTRICA";"1"
|
||||
"193";"INLOCUIRE BARA PROTECTIE FATA";"1"
|
||||
"136";"D/R ROATA SPATE DR AXA 4";"1"
|
||||
"135";"D/R ROATA DR SPATE";"1"
|
||||
"137";"D/R ROTI ( AXA 1/2/3) SI VERIFICAT ELEMENTE DE FRANARE";"1"
|
||||
"141";"D/R SCAUN SOFER";"1"
|
||||
"140";"D/R SCARA USA STG";"1"
|
||||
"134";"D/R ROATA";"1"
|
||||
"125";"D/R POMPA DE APA";"1"
|
||||
"124";"D/R PARASOC STG BARA FATA";"1"
|
||||
"126";"D/R PRAG STG";"1"
|
||||
"130";"D/R RADIATOR APA SI VERIFICAT ETANSEITATE";"1"
|
||||
"128";"D/R PROIECTOR FATA DR";"1"
|
||||
"150";"ECHIPAT ANEXE PE MOTOR";"1"
|
||||
"148";"D/R TURBOSUFLANTA";"1"
|
||||
"154";"EFECTUAT REGLAJ DIRECTIE";"1"
|
||||
"162";"EFECTUAT TEST FRANARE";"1"
|
||||
"161";"EFECTUAT REVIZIE PERIODICA-MICA";"1"
|
||||
"147";"D/R TURBINA";"1"
|
||||
"143";"D/R SI INLOCUIRE SEMNALIZATOR FAT DR";"1"
|
||||
"142";"D/R SEMNALIZARE LATERALA DR";"1"
|
||||
"144";"D/R SUPORT STG BARA FATA";"1"
|
||||
"146";"D/R TREAPTA ARIPA DR CABINA";"1"
|
||||
"145";"D/R SUSPENSIE FATA";"1"
|
||||
"202";"INLOCUIT ALM";"1"
|
||||
"242";"INLOCUIT BEC FAR ST + DR";"1"
|
||||
"241";"INLOCUIT BEC FAR DR";"1"
|
||||
"243";"INLOCUIT BEC FAZA SCURTA";"1"
|
||||
"245";"INLOCUIT BEC POZITIE";"1"
|
||||
"244";"INLOCUIT BEC LAMPA NR";"1"
|
||||
"240";"INLOCUIT BCM";"1"
|
||||
"236";"INLOCUIT BASCULA STG";"1"
|
||||
"232";"INLOCUIT BARA FATA DR";"1"
|
||||
"237";"INLOCUIT BASCULA STG FATA";"1"
|
||||
"239";"INLOCUIT BASCULE FATA";"1"
|
||||
"238";"INLOCUIT BASCULA SUPERIOARA FATA STG";"1"
|
||||
"256";"INLOCUIT BORNA BATERIE MINUS";"1"
|
||||
"254";"INLOCUIT BECURI PE LAMPILE GABARIT";"1"
|
||||
"257";"INLOCUIT BRATE SUSPENSIE FATA";"1"
|
||||
"259";"INLOCUIT BUCSI ARCURI FATA";"1"
|
||||
"258";"INLOCUIT BROASCA USA FATA STG";"1"
|
||||
"252";"INLOCUIT BECURI LAMPI GABARIT";"1"
|
||||
"247";"INLOCUIT BEC STOP";"1"
|
||||
"246";"INLOCUIT BEC SEMNALIZARE SI POZITIE";"1"
|
||||
"248";"INLOCUIT BEC STOP FRANA";"1"
|
||||
"251";"INLOCUIT BECURI FARURI FATA";"1"
|
||||
"250";"INLOCUIT BECURI FARURI";"1"
|
||||
"212";"INLOCUIT AMORTIZOR CABINA FATA STG/DR";"1"
|
||||
"211";"INLOCUIT AMORTIZOR AXA FATA (AMBELE PARTI)";"1"
|
||||
"213";"INLOCUIT AMORTIZOR DR FATA";"1"
|
||||
"215";"INLOCUIT ANSAMBLU BASCULA FATA STG";"1"
|
||||
"214";"INLOCUIT ANSAMBLU BASCULA FATA DR";"1"
|
||||
"210";"INLOCUIT AMORTIZOARE SPATE";"1"
|
||||
"205";"INLOCUIT AMORTIOARE FATA";"1"
|
||||
"204";"INLOCUIT AMBREIAJ";"1"
|
||||
"207";"INLOCUIT AMORTIZOARE CABINA FATA";"1"
|
||||
"209";"INLOCUIT AMORTIZOARE FATA S + D";"1"
|
||||
"208";"INLOCUIT AMORTIZOARE FATA";"1"
|
||||
"228";"INLOCUIT BALAMALE USA STG FATA";"1"
|
||||
"227";"INLOCUIT BALAMALE USA DR";"1"
|
||||
"229";"INLOCUIT BARA DRT. FATA";"1"
|
||||
"231";"INLOCUIT BARA FATA + SUPORTI BARA";"1"
|
||||
"230";"INLOCUIT BARA FATA";"1"
|
||||
"226";"INLOCUIT ARIPA ROATA FATA STG";"1"
|
||||
"220";"INLOCUIT ARC SPATE STG";"1"
|
||||
"216";"INLOCUIT ANSAMBLU USCATOR AER";"1"
|
||||
"221";"INLOCUIT ARC SPATE(O PARTE)";"1"
|
||||
"225";"INLOCUIT ARIPA ROATA FATA DR";"1"
|
||||
"223";"INLOCUIT ARIPA + ORNAMENT STG. PUNTE FATA";"1"
|
||||
"123";"D/R ORNAMENT STG GRILA";"1"
|
||||
"36";"DEMONTAT FURCI CUPLARE CV";"1"
|
||||
"35";"DEMONTAT COMPRESOR A/C";"1"
|
||||
"37";"DEMONTAT REZERVOR COMBUSTIBIL";"1"
|
||||
"40";"DEMONTAT SEGMENTI CILINDRU EXPULZIE";"1"
|
||||
"38";"DEMONTAT ROTI AXA SPATE";"1"
|
||||
"34";"DEMONTAT BORD SI INLOCUIT CEASURI";"1"
|
||||
"29";"DEMONTAT ANEXE DE PE MOTOR";"1"
|
||||
"28";"DEMNTAT BUTUC AXA SPATE DR";"1"
|
||||
"30";"DEMONTAT ARC AXA SPATE STG";"1"
|
||||
"33";"DEMONTAT BORD DREAPTA";"1"
|
||||
"32";"DEMONTAT BORD AUTO";"1"
|
||||
"50";"DEZANSAMBLAT ARC AXA SPATE STG";"1"
|
||||
"48";"DESFACUT SI REPARAT CILINDRI COMPACTARE";"1"
|
||||
"52";"DIAGNOZA NOX";"1"
|
||||
"54";"DIAGNOZA PTO";"1"
|
||||
"53";"DIAGNOZA: MARTOR CHECK ENGINE";"1"
|
||||
"47";"DESCARCAT-INCARCAT INST A/C";"1"
|
||||
"42";"DEMONTAT/MONTAT CAPAC BAIE ULEI";"1"
|
||||
"41";"DEMONTAT SISTEM EVACUARE";"1"
|
||||
"43";"DEMONTAT/MONTAT CILINDRU EXPULZIE";"1"
|
||||
"46";"DEPRESAT RULMENTI";"1"
|
||||
"45";"DEMONTAT/MONTAT CV PE AUTO";"1"
|
||||
"10";"COMPLETAT CU FREON";"1"
|
||||
"9";"COMPLETARE LICHID FRANA";"1"
|
||||
"11";"COMPLETAT INSTALATIE FREON";"1"
|
||||
"13";"CURATAT FILTRU DE PARTICULE";"1"
|
||||
"12";"CURATAT BIELA SI INLOCUIT CUZINETI (6 BIELE)";"1"
|
||||
"7";"APARATOARE ARIPA";"1"
|
||||
"2";"AERISIT INSTALATIE FRANARE";"1"
|
||||
"1";"AERISIT FRANE";"1"
|
||||
"3";"AERISIT INSTALATIE HIDRAULICA";"1"
|
||||
"6";"ANSAMBLAT BORD AUTO";"1"
|
||||
"5";"AERISIT SISTEM HIDRAULIC AMBREIAJ";"1"
|
||||
"23";"CURATAT PINI";"1"
|
||||
"22";"CURATAT MUFE SI CABLAJE";"1"
|
||||
"24";"CURATAT PISTOANE SI INLOCUIT SEGMENTI ( 6 PISTOANE)";"1"
|
||||
"27";"DECONECTAT CONDUCTE COMBUSTIBIL";"1"
|
||||
"26";"CURATAT SUPAPA DEBIT GAZE";"1"
|
||||
"21";"CURATAT INTRECOOLER";"1"
|
||||
"17";"CURATAT INSTALATIE DE AER";"1"
|
||||
"15";"CURATAT INSTALATIE ADBLUE";"1"
|
||||
"18";"CURATAT INSTALATIE DRENARE A/C";"1"
|
||||
"20";"CURATAT INSTALATIE RACIRE";"1"
|
||||
"19";"CURATAT INSTALATIE ELECTRICA PORNIRE";"1"
|
||||
"56";"DIAGNOZA SI UPDATE AUTO";"1"
|
||||
"105";"D/R DISC FRANA SI BUTUC ROATA DR SPATE";"1"
|
||||
"102";"D/R CUTIE VITEZE";"1"
|
||||
"106";"D/R EPURATOR GAZE SI INLOCUIT RACORD";"1"
|
||||
"109";"D/R FILTRU DE PARTICULE SI INLOCUIT GARNITURI + COLIERE";"1"
|
||||
"108";"D/R FATA DE USA STG";"1"
|
||||
"97";"D/R CONDUCTA PRESIUNE POMPA SERVO";"1"
|
||||
"93";"D/R CILINDRU DE EXPULZIE";"1"
|
||||
"92";"D/R CILINDRI HIDRAULICI SPATE COMPACTOR";"1"
|
||||
"94";"D/R CILINDRU EXPULZIE";"1"
|
||||
"96";"D/R COMPONENTE CATALIZATOR";"1"
|
||||
"95";"D/R COLTAR FATA DR";"1"
|
||||
"117";"D/R LAMPA SEMNALIZARE DR";"1"
|
||||
"116";"D/R KIT AMBREIAJ + VOLANTA";"1"
|
||||
"118";"D/R MANER CAPOTA MOTOR";"1"
|
||||
"120";"D/R MOTOR";"1"
|
||||
"119";"D/R MODUL ALIMENTARE AD-BLUE";"1"
|
||||
"115";"D/R JANTA FATA STG";"1"
|
||||
"111";"D/R GEAM INF USA DR";"1"
|
||||
"110";"D/R FILTRU PARTICULE";"1"
|
||||
"112";"D/R GRILA RADIATOR";"1"
|
||||
"114";"D/R JANTA FATA DR";"1"
|
||||
"113";"D/R INJECTOARE";"1"
|
||||
"66";"D/R + REPARAT ELECTROMOTOR";"1"
|
||||
"65";"D/R RADIATOARE";"1"
|
||||
"67";"D/R ANVELOPA FATA DR";"1"
|
||||
"69";"D/R APARATOARE";"1"
|
||||
"68";"D/R ANVELOPA FATA STG";"1"
|
||||
"64";"D/M SI REPARAT CILINDRU LIFT";"1"
|
||||
"60";"D/M FILTRU PARTICULE";"1"
|
||||
"59";"D/M CHIULASA MOTOR SI INLOCUIT GARNITURA CHIULASA";"1"
|
||||
"61";"D/M GRUP DIFERENTIAL";"1"
|
||||
"63";"D/M SI REPARAT CILINDRU GHEARA";"1"
|
||||
"62";"D/M SI REPARAT CILINDRU EXPULZIE";"1"
|
||||
"86";"D/R CAPITONAJ USA DR SP\";"1"
|
||||
"85";"D/R CAPITONAJ FATA DR CABINA";"1"
|
||||
"88";"D/R CATALIZATOR X 2";"1"
|
||||
"91";"D/R CHIULOASA MOTOR";"1"
|
||||
"89";"D/R CAUTATOR CV";"1"
|
||||
"84";"D/R CAPAC SUPERIOR OGLINDA EXT DR";"1"
|
||||
"74";"D/R BARA FATA DR SI INLOCUIT";"1"
|
||||
"73";"D/R BARA FATA DR";"1"
|
||||
"79";"D/R BUTUC ROATA SPATE";"1"
|
||||
"82";"D/R CADRU MOTOR FATA";"1"
|
||||
"80";"D/R BUTUCI ROATA SPATE";"1"
|
||||
"260";"INLOCUIT BUCSI ARCURI SPATE + BRIDE";"1"
|
||||
"418";"INLOCUIT FURTUN RACIRE COMPRESOR";"1"
|
||||
"417";"INLOCUIT FURTUN INTERCOOLER";"1"
|
||||
"419";"INLOCUIT FURTUNE INSTALATIE HIDRAULICA";"1"
|
||||
"421";"INLOCUIT FUZETA FATA DR";"1"
|
||||
"420";"INLOCUIT FURTUNE RACIRE";"1"
|
||||
"416";"INLOCUIT FURTUN INFERIOR RADIATOR APA";"1"
|
||||
"412";"INLOCUIT FOAIE 2 ARC SPATE STG";"1"
|
||||
"411";"INLOCUIT FOAIE 2 ARC SPATE DR";"1"
|
||||
"413";"INLOCUIT FOAIE 3 ARC SPATE";"1"
|
||||
"415";"INLOCUIT FURTUN COMPRESOR AER";"1"
|
||||
"414";"INLOCUIT FULIE ALTERNATOR";"1"
|
||||
"430";"INLOCUIT GEAM OGLINDA MICA DR";"1"
|
||||
"429";"INLOCUIT GEAM OGLINDA MIC STG";"1"
|
||||
"431";"INLOCUIT GIROFAR FATA";"1"
|
||||
"436";"INLOCUIT INTINZATOR ALTERNATOR";"1"
|
||||
"432";"INLOCUIT GRILA SUPERIOARA";"1"
|
||||
"428";"INLOCUIT GARNITURI RACITOR DE ULEI MOTOR";"1"
|
||||
"423";"INLOCUIT GARNITURA CHIULOASA";"1"
|
||||
"422";"INLOCUIT GARNITURA CAPAC CUVA";"1"
|
||||
"425";"INLOCUIT GARNITURI EGR";"1"
|
||||
"427";"INLOCUIT GARNITURI FILTRU";"1"
|
||||
"426";"INLOCUIT GARNITURI ETANSARE DPF";"1"
|
||||
"386";"INLOCUIT ETRIER STG";"1"
|
||||
"385";"INLOCUIT ETRIER FRANA DR SPATE";"1"
|
||||
"387";"INLOCUIT FAR DR";"1"
|
||||
"389";"INLOCUIT FAR DR COMPLET";"1"
|
||||
"388";"INLOCUIT FAR DR + LAMPA POZITIE";"1"
|
||||
"384";"INLOCUIT ELECTROVALVA";"1"
|
||||
"380";"INLOCUIT DOZATOR ADBLUE";"1"
|
||||
"379";"INLOCUIT DISTRIBUITOR LIFT";"1"
|
||||
"381";"INLOCUIT ECU GEARBOX";"1"
|
||||
"383";"INLOCUIT ECU ULEI MOTOR";"1"
|
||||
"382";"INLOCUIT ECU PTO";"1"
|
||||
"404";"INLOCUIT FILTRU POLEN";"1"
|
||||
"401";"INLOCUIT FILTRU DPF";"1"
|
||||
"405";"INLOCUIT FILTRU SCRUF";"1"
|
||||
"408";"INLOCUIT FOAIE DE ARC FATA";"1"
|
||||
"406";"INLOCUIT FILTRU ULEI";"1"
|
||||
"400";"INLOCUIT FILTRU DE ULEI";"1"
|
||||
"392";"INLOCUIT FERODOURI AXA 2 FATA STG/DR";"1"
|
||||
"391";"INLOCUIT FAR STG";"1"
|
||||
"393";"INLOCUIT FILTRE MOTORINA";"1"
|
||||
"398";"INLOCUIT FILTRU DE AER MOTOR";"1"
|
||||
"395";"INLOCUIT FILTRU ADBLUE+CAPAC FILTRU ADBLUE";"1"
|
||||
"437";"INLOCUIT KIT ACCESORII";"1"
|
||||
"491";"INLOCUIT OGLINDA STG";"1"
|
||||
"488";"INLOCUIT NUCA SCHIMBATOR";"1"
|
||||
"493";"INLOCUIT OGLINDA DR";"1"
|
||||
"496";"INLOCUIT OGLINDA SI BUSON REZERVOR";"1"
|
||||
"494";"INLOCUIT OGLINDA MICA STG.";"1"
|
||||
"487";"INLOCUIT MOTOR ELECTRIC ACTIONARE LIFT";"1"
|
||||
"482";"INLOCUIT MODUL DOZARE UPSTREAM CATALIZATOR";"1"
|
||||
"481";"INLOCUIT MODUL - POMPA ADBLUE";"1"
|
||||
"483";"INLOCUIT MODULATOR ABS/EBS";"1"
|
||||
"486";"INLOCUIT MONITOR+CAMERA VIDE";"1"
|
||||
"485";"INLOCUIT MONITOR LED";"1"
|
||||
"508";"INLOCUIT PINION VIT2 + SINCROANE 1/2";"1"
|
||||
"506";"INLOCUIT PINION CV VIT. 6";"1"
|
||||
"509";"INLOCUIT PIVOTI";"1"
|
||||
"516";"INLOCUIT PLACUTE FR";"1"
|
||||
"513";"INLOCUIT PLACA DE COMANDA LIFT";"1"
|
||||
"505";"INLOCUIT PINION CV VIT 2";"1"
|
||||
"498";"INLOCUIT OGLINDA STG";"1"
|
||||
"497";"INLOCUIT OGLINDA ST";"1"
|
||||
"500";"INLOCUIT ORNAMENT ARIPA DRT. PUNTE FATA";"1"
|
||||
"504";"INLOCUIT PERNA AER";"1"
|
||||
"503";"INLOCUIT ORNAMENT PROIECTOR";"1"
|
||||
"458";"INLOCUIT LAMPA NUMAR";"1"
|
||||
"452";"INLOCUIT LAMPA DR PE CABINA";"1"
|
||||
"459";"INLOCUIT LAMPA SEMNAL PE OGLINDA DREAPA";"1"
|
||||
"461";"INLOCUIT LAMPA SEMNALIZARE STG PE OGLINDA";"1"
|
||||
"460";"INLOCUIT LAMPA SEMNAL STANGA IN BARA";"1"
|
||||
"451";"INLOCUIT LAMPA CORN STG SPATE";"1"
|
||||
"442";"INLOCUIT KIT CUREA ACCESORII";"1"
|
||||
"438";"INLOCUIT KIT AMBREAJ";"1"
|
||||
"445";"INLOCUIT KIT GARNITURI CILINDRU EXPULZIE";"1"
|
||||
"449";"INLOCUIT LAMPA CABINA DR";"1"
|
||||
"446";"INLOCUIT KIT HIDRAULIC";"1"
|
||||
"476";"INLOCUIT MANER USA DR FATA";"1"
|
||||
"474";"INLOCUIT MACARA GEAM USA FATA STG";"1"
|
||||
"477";"INLOCUIT MANER USA FATA STG";"1"
|
||||
"480";"INLOCUIT MANSETE LA CILINDRU DE EXPULZIE";"1"
|
||||
"479";"INLOCUIT MANETA SEMNALIZARE";"1"
|
||||
"473";"INLOCUIT MACARA GEAM STG";"1"
|
||||
"465";"INLOCUIT LAMPI";"1"
|
||||
"463";"INLOCUIT LAMPA SPATE DRT.";"1"
|
||||
"470";"INLOCUIT LICHID RACIRE";"1"
|
||||
"472";"INLOCUIT MACARA GEAM DR FATA";"1"
|
||||
"471";"INLOCUIT MACARA CU MOTORAS USA STG";"1"
|
||||
"378";"INLOCUIT DISTRIBUITOR HIDRAULIC PRINCIPAL";"1"
|
||||
"298";"INLOCUIT CAPETE BARA AXA VIRATOARE S +D";"1"
|
||||
"297";"INLOCUIT CAPAC VIZITARE FAR STG SUPERIOR";"1"
|
||||
"299";"INLOCUIT CAPETE BARA ST + DR";"1"
|
||||
"301";"INLOCUIT CAPETE DE BARA LA CASETA DE DIRECTIE";"1"
|
||||
"300";"INLOCUIT CAPETE BARA TRANSVERSALA S + D AXA 1";"1"
|
||||
"296";"INLOCUIT CAPAC VIZITARE FAR STG INFERIOR";"1"
|
||||
"287";"INLOCUIT CABLURI FRANA PARCARE";"1"
|
||||
"286";"INLOCUIT CABLURI FRANA MANA";"1"
|
||||
"289";"INLOCUIT CAMERA SI MONITOR";"1"
|
||||
"293";"INLOCUIT CAP BARA DIRECTIE ST + DR";"1"
|
||||
"292";"INLOCUIT CAP BARA (AMBELE PARTI)";"1"
|
||||
"310";"INLOCUIT CAUTATOR CV";"1"
|
||||
"309";"INLOCUIT CATALIZATOR";"1"
|
||||
"311";"INLOCUIT CHIULASA COMPRESOR";"1"
|
||||
"313";"INLOCUIT CILINDRU LATERAL";"1"
|
||||
"312";"INLOCUIT CILINDRU AMBREIAJ";"1"
|
||||
"308";"INLOCUIT CASETA DE DIRECTIE";"1"
|
||||
"304";"INLOCUIT CAPOTA MOTOR";"1"
|
||||
"302";"INLOCUIT CAPETE DE BARA S + D";"1"
|
||||
"305";"INLOCUIT CARCASA OGLINDA DR";"1"
|
||||
"307";"INLOCUIT CARDAN";"1"
|
||||
"306";"INLOCUIT CARCASA OGLINDA STG.";"1"
|
||||
"270";"INLOCUIT BUJIE INCANDESCENTA (4)";"1"
|
||||
"269";"INLOCUIT BUCSI ETRIERI FATA";"1"
|
||||
"271";"INLOCUIT BUJIE MOTOR 1 BUCATA";"1"
|
||||
"274";"INLOCUIT BUJII 2 BUCATI";"1"
|
||||
"272";"INLOCUIT BUJII";"1"
|
||||
"268";"INLOCUIT BUCSI CABINA SPATE";"1"
|
||||
"262";"INLOCUIT BUCSI BARA STABILIZATIOARE FATA";"1"
|
||||
"261";"INLOCUIT BUCSI BARA STAB SPATE";"1"
|
||||
"263";"INLOCUIT BUCSI BARA STABILIZATOARE FATA";"1"
|
||||
"266";"INLOCUIT BUCSI BASCULA FATA STG + DR";"1"
|
||||
"265";"INLOCUIT BUCSI BARA TORSIUNE FATA DR";"1"
|
||||
"282";"INLOCUIT CABLU AMBREIAJ";"1"
|
||||
"281";"INLOCUIT CABLU ACCELERATIE";"1"
|
||||
"283";"INLOCUIT CABLU TIMONERIE";"1"
|
||||
"285";"INLOCUIT CABLURI FRANA DE MANA";"1"
|
||||
"284";"INLOCUIT CABLURI FRANA DE MANA";"1"
|
||||
"280";"INLOCUIT CABLAJ POMPA ADBLUE";"1"
|
||||
"276";"INLOCUIT BUTUC ROATA";"1"
|
||||
"275";"INLOCUIT BUTUC FATA DREAPTA";"1"
|
||||
"277";"INLOCUIT BUTUC ROATA FATA + RULMENT DR";"1"
|
||||
"279";"INLOCUIT CABLAJ NOXE";"1"
|
||||
"278";"INLOCUIT BUTUCI FATA CU RULMENTI";"1"
|
||||
"314";"INLOCUIT CILINDRU RECEPTOR";"1"
|
||||
"354";"INLOCUIT CUREA ALTERNATOR";"1"
|
||||
"353";"INLOCUIT CUREA ALTERBATOR";"1"
|
||||
"355";"INLOCUIT CUREA +ROLA TRANSMISIE";"1"
|
||||
"357";"INLOCUIT CUREA TRANSMISIE";"1"
|
||||
"356";"INLOCUIT CUREA SI ROLE";"1"
|
||||
"352";"INLOCUIT CUREA A/C";"1"
|
||||
"348";"INLOCUIT CRUCI CARDANICE";"1"
|
||||
"347";"INLOCUIT CRUCE CRADAN";"1"
|
||||
"349";"INLOCUIT CULISANTE ETRIER SPATE STG/DR";"1"
|
||||
"351";"INLOCUIT CUREA + ROLA DISTRIBUTIE";"1"
|
||||
"350";"INLOCUIT CUREA + ROLA AC";"1"
|
||||
"367";"INLOCUIT DISCURI + PLACUTE FATA";"1"
|
||||
"366";"INLOCUIT DISC SI PLACUTE AXA FATA";"1"
|
||||
"370";"INLOCUIT DISCURI FATA";"1"
|
||||
"377";"INLOCUIT DISTRIBUITOR COMPACTARE";"1"
|
||||
"371";"INLOCUIT DISCURI FATA + PLACUTE";"1"
|
||||
"365";"INLOCUIT DISC FRINA + PLACUTE SPATE";"1"
|
||||
"361";"INLOCUIT DISC AMBREIAJ";"1"
|
||||
"359";"INLOCUIT CUZINETI BIELA + 1 BIELA";"1"
|
||||
"362";"INLOCUIT DISC FR SP (AMBELE)";"1"
|
||||
"364";"INLOCUIT DISC FRANA(AMBELE)";"1"
|
||||
"363";"INLOCUIT DISC FRANA FATA STG/DR";"1"
|
||||
"322";"INLOCUIT COMPRESOR + CUREA";"1"
|
||||
"321";"INLOCUIT COMANDA LIFT";"1"
|
||||
"323";"INLOCUIT COMPRESOR A.C.";"1"
|
||||
"326";"INLOCUIT COMPRESOR AER";"1"
|
||||
"324";"INLOCUIT COMPRESOR A/C";"1"
|
||||
"320";"INLOCUIT COLIERE TURBO";"1"
|
||||
"316";"INLOCUIT CIRCUIT HIDRAULIC AMBREIAJ";"1"
|
||||
"315";"INLOCUIT CILINDRU RECEPTOR AMBREIAJ";"1"
|
||||
"317";"INLOCUIT CLAPETA ACCELERATIE";"1"
|
||||
"319";"INLOCUIT COLIERE SI GARNITURI EVACUARE";"1"
|
||||
"318";"INLOCUIT COLIERE FURTUN TURBINA";"1"
|
||||
"337";"INLOCUIT CONDUCTA RACIRE EGR";"1"
|
||||
"336";"INLOCUIT CONDUCTA RACIRE COMPRESOR/CHIULOASA";"1"
|
||||
"339";"INLOCUIT CONDUCTA SISTEM INCALZIRE AD-BLUE";"1"
|
||||
"341";"INLOCUIT CONDUCTE ALIMENTARE COMBUSTIBIL";"1"
|
||||
"340";"INLOCUIT CONDUCTA ULEI TURBINA";"1"
|
||||
"335";"INLOCUIT CONDUCTA POMPA RABATARE";"1"
|
||||
"328";"INLOCUIT COMUTATOT STOP FRANA";"1"
|
||||
"327";"INLOCUIT COMUTATOR MI";"1"
|
||||
"329";"INLOCUIT CONDUCT RACIRE VAS - RADIATOR";"1"
|
||||
"334";"INLOCUIT CONDUCTA INJECTOR";"1"
|
||||
"333";"INLOCUIT CONDUCTA EGR";"1"
|
||||
|
430
docs/prd/prd-5.10-ux-filtre-pill-paginare-mapari-meniu.md
Normal file
430
docs/prd/prd-5.10-ux-filtre-pill-paginare-mapari-meniu.md
Normal file
@@ -0,0 +1,430 @@
|
||||
<!-- /autoplan restore point: ~/.gstack/projects/romfast-rar-autopass/main-autoplan-restore-20260625-120049.md -->
|
||||
# PRD 5.10 — UX trimiteri (pill filtre, paginare, detaliu) + Mapari in meniu
|
||||
|
||||
**Stare**: inchis (2026-06-25 — 14 stories + fix US-006b TDD prin echipa; VERIFY PASS; `/code-review high` 1 finding material reparat; regresie 896 passed / 1 skipped / 0 failed; fonturi IBM Plex reale)
|
||||
|
||||
> Proces complet: `docs/ROADMAP.md` §5. Contractul RAR (sursa de adevar): `docs/api-rar-contract.md`.
|
||||
> Starea trece: `draft → aprobat → in-executie → verify-pass → inchis` (actualizata de lead).
|
||||
|
||||
## 1. Obiectiv
|
||||
|
||||
Curatare UX a dashboard-ului pe doua zone: (a) **tabelul de trimiteri** — pill-uri de filtrare
|
||||
pe categorii de problema in locul listei expandate de ID-uri, filtrare pe interval de data care
|
||||
chiar functioneaza pe timestamp-uri, paginare numerotata, VIN sub numarul de inmatriculare,
|
||||
editarea operatiei RAR si afisarea operatiei de service in detaliu, plus erori mai putin verbose
|
||||
in formularul de editare; (b) **Mapari** — mutata in meniul hamburger (fara tab-uri pe pagina
|
||||
principala), o singura pagina cu sectiunile consolidate si butoane de salvare/stergere vizibile.
|
||||
|
||||
Pur UI/UX + un fix de filtrare backend. Backend-ul de trimitere (worker, masina de stari,
|
||||
idempotenta, mapping-rezolvare) ramane **NEATINS**.
|
||||
|
||||
## 2. Non-Goals (anti scope-creep)
|
||||
|
||||
- **Nu** atingem worker-ul, masina de stari, idempotenta sau logica de mapare operatie→cod.
|
||||
- **Nu** permitem editarea operatiei RAR pe stari trimise/in curs (`sent`/`sending`/`queued`) —
|
||||
la RAR `FINALIZATA` e terminal; editarea apare doar pe `needs_data`/`needs_mapping`/`error`.
|
||||
- **Nu** schimbam paginarea client-side existenta din tabelele Mapari (datatable `data-dt`).
|
||||
- **Nu** schimbam mecanismul de persistenta/anti-FOUC al temei (PRD 5.3) — il extindem cu teme noi, nu il rescriem.
|
||||
- **Nu** adaugam filtre noi (doar pill-uri pe categoriile de problema deja existente + fix data).
|
||||
- **Nu** modificam contractul API `POST /v1/prezentari` / `GET /v1/prezentari` (operatia de
|
||||
service afisata vine din payload-ul deja stocat, nu cere camp nou).
|
||||
- **Nu** reproiectam meniul hamburger in sine — doar adaugam intrarea „Mapari" si scoatem tab-bar-ul.
|
||||
- **Acceptat explicit (review C4)**: doua idiomuri de paginare coexista — server-side numerotat pe Trimiteri (US-004) vs client-side `data-dt` (Inapoi/Inainte) pe Mapari. Diferenta e intentionata, nu „de reparat" ulterior.
|
||||
- **Acceptat explicit (review C2)**: butonul ciclic de tema (US-014) are cost de descoperire (4 teme fara optiuni vizibile). Compensat prin `aria-label` curenta+urmatoarea + tooltip; nu trecem la meniu/popover in aceasta livrabila.
|
||||
|
||||
## 3. Stories atomice
|
||||
|
||||
> Fiecare story: cea mai mica unitate care lasa sistemul functional. Backend + UI pentru acelasi
|
||||
> comportament = 2 stories.
|
||||
|
||||
### US-001: Fix filtrare pe interval de data (backend)
|
||||
**Ca** operator **vreau** ca filtrul „Data de la / pana la" sa returneze toate trimiterile din
|
||||
acea perioada **pentru ca** acum, fiindca `data_prestatie` poate avea ora/minut/secunda,
|
||||
comparatia de string exclude randurile si tabelul apare gol.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/routes.py` (`fragment_submissions`, `_is_iso_date`), `tests/test_web_filtre_submissions.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_filtre_submissions.py` — `test_filtru_data_include_timestamp_cu_ora`, `test_filtru_data_interval_inclusiv_capete`, `test_filtru_data_ignora_valori_ne_data`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **(must-fix, review C1)** Cauza reala = garda `_is_iso_date` cere `len(s)==10` (`routes.py:640-649`), deci orice `data_prestatie` cu ora e **exclusa inainte** de comparatie. Fix: introdu `_iso_date_prefix(value) -> str | None` (intoarce `value[:10]` daca parseaza ca `YYYY-MM-DD`, altfel `None`) si foloseste-l **atat** pentru garda **cat si** pentru comparatie. Nu modifica doar liniile de comparatie — altfel bug-ul ramane.
|
||||
- [ ] Filtrarea compara doar **portiunea de data** (primele 10 caractere, `YYYY-MM-DD`) a lui `data_prestatie`, chiar daca valoarea contine ora/minut/secunda (ex. `2026-06-20 14:35:07` sau `2026-06-20T14:35:07`).
|
||||
- [ ] Intervalul e **inclusiv la ambele capete**: `data_de <= data(rand) <= data_pana`.
|
||||
- [ ] O singura limita (doar `data_de` sau doar `data_pana`) functioneaza corect.
|
||||
- [ ] Valorile care nu incep cu o data ISO valida raman excluse din filtru (comportament actual pastrat).
|
||||
- [ ] `python3 -m pytest tests/test_web_filtre_submissions.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — import/seed cu o trimitere cu `data_prestatie` ce contine ora; filtru pe acea zi → randul apare.
|
||||
|
||||
### US-002: Expune operatia de service in view-ul de rand si detaliu (backend)
|
||||
**Ca** operator **vreau** sa vad operatia de service originala (codul intern / denumirea venita
|
||||
prin API sau import CSV) **pentru ca** vreau sa stiu ce a cerut service-ul, nu doar codul RAR mapat.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/payload_view.py` (sau `app/web/routes.py` `_detaliu_ctx`/`prezentare_din_payload`), `tests/test_payload_view.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_payload_view.py` — `test_operatie_service_din_cod_op_service`, `test_operatie_service_din_denumire`, `test_fara_operatie_service_cand_lipseste`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **(must-fix, review M1)** `prezentare_din_payload` colapseaza acum service-op si RAR-op in chei suprapuse (`operatie = denumire or cod`, `cod_rar = cod_prestatie`; `payload_view.py:111-113`). Adauga **chei noi distincte** (ex. `op_service_cod` + `op_service_denumire`) din `cod_op_service`/`denumire`, separate de operatia RAR mapata.
|
||||
- [ ] **Conventie goala (must-fix, review M1)**: payload_view emite `EMPTY="—"` pentru valori lipsa (`payload_view.py:119-127`), dar US-007 cere „randul nu apare deloc (fara — gol)". Alege O conventie: fie campul nou intoarce `""`/`None` (nu `—`), fie template-ul testeaza fata de `'—'` (ca randul VIN, `_submissions.html:63`). Documenteaz-o aici si in US-007.
|
||||
- [ ] Cand payload-ul nu contine operatie de service (a venit direct cu `cod_prestatie`), campul e gol conform conventiei de mai sus, fara a arunca.
|
||||
- [ ] Helper-ul ramane pur (fara DB), defensiv la payload lipsa/corupt.
|
||||
- [ ] `python3 -m pytest tests/test_payload_view.py -q` trece.
|
||||
- **Verificare E2E**: `POST /v1/prezentari` cu `cod_op_service`+`denumire` → randul are operatia de service in context.
|
||||
|
||||
### US-003: Pill-uri de filtrare pe categorii de problema (UI)
|
||||
**Ca** operator **vreau** pill-uri „Date incomplete / Lipsa cod / Eroare" cu numar in sectiunea de
|
||||
filtrare, in loc de lista expandata `#40 TMB...3456 / B28ERR ...si inca 3` **pentru ca** lista de
|
||||
ID-uri e zgomotoasa; vreau sa apas un pill ca sa filtrez direct pe acea categorie.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_coada.html`, `app/web/templates/_status.html`, `app/web/routes.py` (counts per categorie pe fragmentul de filtre), `tests/test_web_pill_filtre.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_pill_filtre.py` — `test_pill_per_categorie_cu_numar`, `test_pill_click_seteaza_status`, `test_fara_lista_id_uri`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] In sectiunea de filtrare apar pill-uri: **Date incomplete** (`needs_data`), **Lipsa cod** (`needs_mapping`), **Eroare** (`error`), fiecare cu numarul total scoped pe cont; pill-ul lipseste/e ascuns cand numarul e 0.
|
||||
- [ ] **(must-fix, review M3/design-H1)** Lista de ID-uri si blocul `blocate_actionabil` sunt in **`_status.html:68-77`** (nu `_coada.html`), generate de `_blocate_actionabil` (`routes.py:562-594`). La eliminarea ID-urilor, scoate si codul mort care calcula `prezentare_din_payload`/`vin_partial` per rand. Pill-urile reutilizeaza contoarele deja calculate (`_status_counts`/`n`) — fara filtru backend nou.
|
||||
- [ ] **(must-fix, review design-M1)** Decide explicit soarta intregului bloc „Necesita atentia ta" + a contorului „Blocate": pill-urile **inlocuiesc** link-urile de categorie (nu triplu-encoda aceeasi informatie cu pill + link + dropdown).
|
||||
- [ ] **(must-fix, review M3/A3)** Pill-urile sunt elemente focalizabile reale (`<button>`/`<a>`, nu `<span onclick>`), cu `focus-visible`, activare Enter/Space, si stare activa via `aria-pressed` (nu doar culoare).
|
||||
- [ ] Click pe un pill filtreaza tabelul pe acea stare (seteaza `status=` pe `fragment_submissions`, deja suportat `routes.py:699`); pill activ evidentiat; click pe activ revine la „toate".
|
||||
- [ ] **(must-fix, review M3)** Sincronizare cu dropdown-ul `status` din `_coada.html:36-44`: ori dropdown-ul reflecta pill-ul activ, ori e eliminat — fara desync (click pill vs valoare dropdown stale).
|
||||
- [ ] Exista si un control „Toate" (toggle) care reseteaza filtrul de categorie. Stare „toate zero" (steady-state sanatos): definita explicit (rand ascuns vs afisaj muted „nicio problema"), nu gap neexplicat.
|
||||
- [ ] **(must-fix, review S1/A5)** Matrice de stil pill rezolvata fara contradictie rosu/galben vs accent: inactiv = contur/text pe culoarea categoriei (`--err`/`--warn`); activ = umplere pe **culoarea categoriei** (NU accent albastru — altfel pill rosu „Date incomplete" devine albastru cand e activ). Contrast text pe pill-ul activ verificat AA in toate cele 3 teme.
|
||||
- [ ] `python3 -m pytest tests/test_web_pill_filtre.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — pill-uri cu numere; click pe „Date incomplete" → tabel filtrat; click din nou → toate.
|
||||
|
||||
### US-004: Paginare numerotata pe tabelul de trimiteri (backend + UI)
|
||||
**Ca** operator **vreau** paginare numerotata pe tabelul de trimiteri **pentru ca** acum se
|
||||
incarca max 200 randuri fara navigare si nu pot ajunge la cele mai vechi.
|
||||
|
||||
- **Depinde de**: US-001 (filtrul de data corect intra in acelasi handler)
|
||||
- **Fisiere**: `app/web/routes.py` (`fragment_submissions`: param `page`, total + slice ramificat — vezi C1), `app/web/templates/_submissions.html`, `app/web/templates/_coada.html` (poll `hx-include` + `page`), `tests/test_web_paginare_submissions.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_paginare_submissions.py` — `test_pagina_implicita_25`, `test_pagina_2_offset`, `test_total_si_numar_pagini`, `test_paginarea_pastreaza_filtrele`, `test_pagina_peste_total_revine_la_ultima`, `test_poll_pastreaza_pagina`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **(must-fix, review H1)** Numararea totalului ramifica dupa tipul de filtru: **fara** filtru Python (status-only / niciun filtru) → SQL `COUNT(*)` + `LIMIT 25 OFFSET (page-1)*25`; **cu** filtru vehicul/data activ → fetch-all (fara LIMIT, ca azi) → filtreaza in Python → `total=len(filtrat)` → slice `[offset:offset+25]`. SQL `COUNT(*)/LIMIT/OFFSET` e **gresit** cand filtrul Python e activ. Plafonul de 200 randuri din bucla se inlocuieste cu „fetch-all-then-slice" pe calea filtrata (altfel paginile >8 dispar silentios).
|
||||
- [ ] Marime pagina fixa **25** randuri; raspunsul include numarul total si pagina curenta.
|
||||
- [ ] **(must-fix, review H2)** `page` in afara intervalului se clampeaza la `[1, ceil(total/25)]` (nu pagina goala); schimbarea unui filtru reseteaza `page` la 1.
|
||||
- [ ] Controale: `‹ Inapoi`, numere de pagina, `Inainte ›`; pagina curenta evidentiata cu `aria-current="page"`; capetele dezactivate (`disabled`) la prima/ultima pagina; pager-ul e **ascuns** cand `pages<=1` sau `total==0` (empty state inlocuieste tabelul). „afiseaza X-Y din N" intr-o regiune `aria-live="polite"`.
|
||||
- [ ] Schimbarea paginii **pastreaza** filtrele active (status/pill, vehicul, data_de, data_pana) — link-urile de paginare poarta toti parametrii de filtru curenti.
|
||||
- [ ] **(must-fix, review L2)** Poll-ul de 15s (`_coada.html` `hx-include="#filtre-trimiteri"`) NU trebuie sa reseteze pagina: include `page` curent in include-ul de poll (hidden input actualizat de paginare) SAU documenteaza explicit reset-pe-poll. Test: `test_poll_pastreaza_pagina`.
|
||||
- [ ] `python3 -m pytest tests/test_web_paginare_submissions.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` cu >25 trimiteri — navigare intre pagini, filtru aplicat ramane la schimbarea paginii, poll-ul nu te scoate de pe pagina curenta.
|
||||
|
||||
### US-005: VIN sub numarul de inmatriculare in tabel (UI)
|
||||
**Ca** operator **vreau** ca VIN-ul sa apara pe rand propriu sub numarul de inmatriculare **pentru
|
||||
ca** pe aceeasi linie e ingramadit si greu de citit.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_submissions.html`, `tests/test_web_submissions_layout.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_submissions_layout.py` — `test_vin_pe_rand_separat_sub_nr`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **(nota, review L1)** VIN-ul e deja randat in `_submissions.html:61-65`, dar **inline** (`<span>` dupa nr, aceeasi celula). Schimbarea e mica: element block sub nr (nu structura noua). Testul asserteaza un element **block-level**, nu doar prezenta.
|
||||
- [ ] In coloana Vehicul, numarul de inmatriculare e pe primul rand; VIN-ul (sau partiala VIN) apare **dedesubt**, in stil muted, nu pe aceeasi linie.
|
||||
- [ ] Cand VIN-ul lipseste, nu apare rand gol (garda `!= '—'` exista deja, `_submissions.html:63`).
|
||||
- [ ] Layout-ul ramane fara scroll orizontal pe tabel (scopat `.tabel-trimiteri`, consistent cu PRD 5.8).
|
||||
- [ ] `python3 -m pytest tests/test_web_submissions_layout.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — VIN sub numar in fiecare rand.
|
||||
|
||||
### US-006: Editare operatie RAR in formularul de detaliu (UI + backend corectie)
|
||||
**Ca** operator **vreau** sa pot schimba operatia RAR (`cod_prestatie`) din formularul de detaliu pe
|
||||
trimiterile blocate **pentru ca** uneori codul mapat e gresit si vreau sa-l corectez inainte de re-trimitere.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_trimitere_detaliu.html`, `app/web/routes.py` (`post_corectie_trimitere` ~979-1124), `tests/test_web_editare_op_rar.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_editare_op_rar.py` — `test_editabil_arata_select_cod_rar`, `test_salvare_schimba_cod_si_repune_in_coada`, `test_idempotency_key_se_schimba`, `test_cod_invalid_respins`, `test_sent_nu_arata_select`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Pe `needs_data`/`needs_mapping`/`error`, formularul afiseaza un **select cu codurile din nomenclator** pentru operatia RAR, pre-selectat pe codul curent.
|
||||
- [ ] **(must-fix, review H3)** Riscul „refoloseste exact calea existenta" e GRESIT formulat: `post_corectie_trimitere` aplica DOAR campurile vehiculului (`vin/nr/data/odometru`), nu citeste niciun cod de prestatie (`routes.py:1009-1012`). Story-ul cere logica **noua** in handler: (a) camp nou `cod_prestatie` in form; (b) validare fata de nomenclator (oglindeste `routes.py:953-961`); (c) injectare in `content["prestatii"][i]["cod_prestatie"]` inainte de `resolve_prestatii`. DUPA injectare, restul caii existente (canonicalize → `build_key` → check coliziune idempotency → re-queue, `routes.py:1028-1104`) recalculeaza corect payload+cheie (`build_key` hashuieste `cod_prestatie`, `idempotency.py:34`).
|
||||
- [ ] Codul nou e validat fata de nomenclator; un cod necunoscut e **respins** (nu se injecteaza, nu se trimite raw — vezi invariantul `cod_prestatie` validat la ingestie din CLAUDE.md).
|
||||
- [ ] Test `test_idempotency_key_se_schimba`: cheia de idempotency **difera** dupa schimbarea codului (nu doar `status=queued`).
|
||||
- [ ] Daca payload-ul are mai multe prestatii, story-ul tinteste prima/operatia editabila explicit (nu ambiguu).
|
||||
- [ ] Pe `sent`/`sending`/`queued` operatia RAR ramane **read-only** (fara select).
|
||||
- [ ] Scoped pe sesiune + CSRF, 404 cross-account.
|
||||
- [ ] `python3 -m pytest tests/test_web_editare_op_rar.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — pe o trimitere `needs_data`, schimba codul RAR din select, salveaza → cod nou aplicat + cheie idempotency noua; cod invalid → respins; pe o trimitere `sent`, codul e read-only.
|
||||
|
||||
### US-007: Afisare operatie de service in detaliu (UI)
|
||||
**Ca** operator **vreau** sa vad operatia de service originala in formularul de detaliu **pentru
|
||||
ca** vreau sa stiu ce a cerut service-ul prin API/CSV, alaturi de codul RAR.
|
||||
|
||||
- **Depinde de**: US-002
|
||||
- **Fisiere**: `app/web/templates/_trimitere_detaliu.html`, `tests/test_web_detaliu_op_service.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_detaliu_op_service.py` — `test_detaliu_arata_operatie_service`, `test_detaliu_omite_cand_lipseste`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] In detaliu apare „Operatie service" (cod intern + denumire) cand a existat in payload, distinct de „Operatie RAR".
|
||||
- [ ] Cand operatia de service lipseste (a venit direct cu `cod_prestatie`), randul nu apare deloc (fara „—" gol).
|
||||
- [ ] Apare atat in contextul editabil cat si in cel read-only.
|
||||
- [ ] `python3 -m pytest tests/test_web_detaliu_op_service.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — trimitere venita cu `cod_op_service` arata operatia de service in detaliu; una venita cu `cod_prestatie` direct nu o arata.
|
||||
|
||||
### US-008: Simplificare eroare in formularul de editare (UI)
|
||||
**Ca** operator **vreau** ca in formularul de editare sa apara doar textul simplu al erorii (subliniat),
|
||||
nu blocul verbose pe 3 niveluri **pentru ca** „De ce / Cum repari" + prefixul tehnic dubleaza un
|
||||
mesaj deja descriptiv (ex. „odometruFinal trebuie sa fie un numar intreg (ca string).").
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_trimitere_detaliu.html` (zona editabila), `tests/test_web_detaliu_eroare_simpla.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_detaliu_eroare_simpla.py` — `test_form_editare_fara_card_3niveluri`, `test_eroare_pe_camp_doar_text_simplu`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **(must-fix, review M4)** Blocul `erori_3n`/`card_erori` e acum randat **inainte** de form, in afara ramurii `{% if editabil %}` (`_trimitere_detaliu.html:22-27`) — deci apare si pe read-only. Ascunde-l DOAR in editare: muta-l/infasoara-l in `{% if not editabil %}` ca sa-l pastrezi pe contextele read-only (catalogul PRD 5.4).
|
||||
- [ ] Eroarea per camp ramane sub input ca **text simplu subliniat** (mesajul descriptiv). Nota: macro-ul `camp` printeaza deja doar `err_map.get(nume)` (mesajul, fara cod de camp) — verifica doar ca `message` nu inglobeaza numele campului.
|
||||
- [ ] **(must-fix, review M6/design)** Erorile **fara camp** (`field is None`) nu trebuie sa dispara silentios cand scoatem cardul 3n: defineste unde apar (ex. un rezumat simplu top-of-form) in formularul de editare.
|
||||
- [ ] Restul contextelor (lista compacta / detaliu read-only) raman neschimbate — simplificarea e scopata doar pe formularul de editare.
|
||||
- [ ] `python3 -m pytest tests/test_web_detaliu_eroare_simpla.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — corectie cu odometru invalid → sub camp apare doar textul erorii subliniat, fara cardul pe 3 niveluri.
|
||||
|
||||
### US-009: Mapari in meniul hamburger + scoatere tab-uri (UI)
|
||||
**Ca** operator **vreau** ca Mapari sa fie o intrare in meniul hamburger, nu tab pe pagina
|
||||
principala **pentru ca** vreau pagina principala curata (doar Acasa), iar tab-bar-ul Acasa/Mapari
|
||||
incurca.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/base.html` (meniu `#cont-menu`), `app/web/templates/dashboard.html` (tab-bar), `app/web/routes.py` (rutare pagina Mapari, ex. `/?tab=mapari` sau `/mapari`), `tests/test_web_mapari_meniu.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapari_meniu.py` — `test_meniu_contine_mapari`, `test_pagina_principala_fara_tabbar_mapari`, `test_ruta_mapari_randeaza_sectiunea`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Meniul hamburger contine o intrare **Mapari** (cu badge pentru `needs_mapping` daca exista contoare). Badge-ul muta din tab-bar pe item-ul de meniu — sursa contorului ramane `badges.mapari` (`routes.py:423-424`); verifica sa nu ramana badge dangling.
|
||||
- [ ] Tab-bar-ul Acasa/Mapari de pe pagina principala e **eliminat**; Acasa devine continutul principal direct.
|
||||
- [ ] **(must-fix, review design-C1)** Scoate si schela ARIA `role="tablist"` orfana + JS-ul de navigare cu sageti din `dashboard.html:16-88` (un `role="tablist"` cu un singur tab e o violare ARIA). Curata plumbing-ul `active_tab` ramas fara sens daca nu mai e folosit.
|
||||
- [ ] Exista o ruta dedicata care randeaza pagina Mapari (server-side, deep-link). `?tab=mapari` ramane valid (`_TABS_VALIDE` include `mapari`, `routes.py:155`; randat de `_render_panel_for_tab`) — nu da 404. Verifica sa nu ramana `hx-get` catre un element de tab eliminat (banner/badge).
|
||||
- [ ] `python3 -m pytest tests/test_web_mapari_meniu.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — fara tab-bar; din ☰ → Mapari → pagina Mapari.
|
||||
|
||||
### US-010: Restructurare pagina Mapari intr-o singura pagina consolidata (UI)
|
||||
**Ca** operator **vreau** o singura pagina Mapari cu „De rezolvat" prima, apoi salvate / reguli text /
|
||||
formate coloane, fara sectiunea de ajutor si fara textul gol **pentru ca** sectiunile separate si
|
||||
textele de ajutor ma incurca.
|
||||
|
||||
- **Depinde de**: US-009
|
||||
- **Fisiere**: `app/web/templates/_mapari.html`, `tests/test_web_mapari_layout.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapari_layout.py` — `test_de_rezolvat_prima`, `test_fara_ajutor_si_empty_text`, `test_ordine_sectiuni`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Ordinea sectiunilor: (1) **De rezolvat** prima, (2) Mapari operatii salvate, (3) Reguli automate (text), (4) Formate de coloane salvate.
|
||||
- [ ] Sectiunea de ajutor (`<details class="ajutor-mapari">`) e **eliminata**.
|
||||
- [ ] Textul empty-state „Nicio operatie nemapata — tot ce a venit s-a tradus in coduri RAR. Importa un fisier nou..." e **eliminat** (sectiunea „De rezolvat" goala nu mai afiseaza acel paragraf).
|
||||
- [ ] Fiecare rand din „De rezolvat" pastreaza select-ul + butonul de salvare vizibil (vezi US-011).
|
||||
- [ ] `python3 -m pytest tests/test_web_mapari_layout.py -q` trece.
|
||||
- **Verificare E2E**: browser pe pagina Mapari — ordine corecta, fara ajutor, fara empty-text.
|
||||
|
||||
### US-012: Branding header „by ROMFAST" + titlu centrat (UI)
|
||||
**Ca** utilizator **vreau** ca header-ul sa aiba branding-ul „by ROMFAST" si titlul pe mijloc
|
||||
**pentru ca** vreau identitate vizuala clara, parte din familia ROMFAST/ROA, nu un header anonim.
|
||||
|
||||
> Sistem de design complet: `DESIGN.md` (sectiunile „Header & branding" + „Culori de brand").
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/base.html` (header + CSS), `tests/test_web_header_branding.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_header_branding.py` — `test_header_contine_by_romfast`, `test_titlu_centrat`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Titlul „Gateway RAR AUTOPASS" e **centrat** in header (grila 3 coloane: controale — tema, versiune, ☰ — la **dreapta**, fara a strica centrarea optica a titlului).
|
||||
- [ ] **(must-fix, review S2)** Plaseaza explicit badge-ul `env` (test/prod, azi `base.html:320`) in grila — celula stanga (echilibru) sau langa titlu — si defineste ordinea de colaps pe mobil (3-col cu titlu centrat + controale care wrap e fragil).
|
||||
- [ ] Sub titlu, mic: wordmark **„by ROMFAST"** redat ca text stilizat — `by` in `--muted`, `ROM` in `#D1342F` (rosu logo), `FAST` in `#2E74D6` (albastru logo). **NU** se foloseste PNG-ul 3D al logo-ului.
|
||||
- [ ] Responsiv: pe mobil wordmark-ul ramane sub titlu, controalele nu se suprapun (degrada elegant).
|
||||
- [ ] Light + dark OK (wordmark pe culori proprii, lizibil pe ambele fundaluri).
|
||||
- [ ] `python3 -m pytest tests/test_web_header_branding.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` desktop + mobil — titlu centrat, „by ROMFAST" cu ROM rosu / FAST albastru, controale la dreapta.
|
||||
|
||||
### US-013: Tema de culori ROMFAST + tipografie (UI)
|
||||
**Ca** utilizator **vreau** o paleta cu accent albastru ROMFAST (ca romfast.ro) si o tipografie
|
||||
coerenta **pentru ca** acum totul e gri si fara identitate, iar produsul e parte din familia ROA.
|
||||
|
||||
> Valori complete: `DESIGN.md` (sectiunile „Decizie cromatica" + „Tipografie").
|
||||
|
||||
- **Depinde de**: design-consultation (DONE — `DESIGN.md` scris)
|
||||
- **Fisiere**: `app/web/templates/base.html` (variabile `:root` + `[data-theme="light"]` + `font-family`), `app/web/static/fonts/` (woff2 IBM Plex), `tests/test_web_tema_culori.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_tema_culori.py` — `test_paleta_accent_azur_definita`, `test_font_ibm_plex_aplicat`, `test_contrast_aa_pe_text_principal`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Accentul devine **azur ROMFAST**: `--accent:#2E74D6` (dark) / `#1F66C9` (light), aplicat prin variabile CSS (butoane primare, pill activ, linkuri, focus) — fara culori noi hardcodate imprastiate.
|
||||
- [ ] Neutrele actualizate conform `DESIGN.md` (dark `--bg:#0f1218`/`--card:#181c24`; light `--bg:#f5f7fa`/`--card:#ffffff`); stari `--ok/--warn/--err` pastrate AA per tema.
|
||||
- [ ] **Tipografie**: `font-family` UI = IBM Plex Sans (fallback `system-ui`); monospace (coduri RAR/VIN/nr.) = IBM Plex Mono. Self-host woff2 cu `font-display:swap` (subset latin-ext pentru diacritice). Fallback de sistem nu strica layout-ul.
|
||||
- [ ] **(must-fix, review M3/design)** `font-display:swap` produce un FOUT system-ui→IBM Plex; pe coloanele tabulare (VIN/coduri, `tabular-nums` `base.html:54`) o nepotrivire de metrici da reflow. Defineste fallback cu metrici ajustate (`size-adjust`/`ascent-override`) SAU accepta explicit FOUT-ul si confirma ca tabularele nu fac reflow vizibil.
|
||||
- [ ] Contrastul textului principal ramane **AA** in ambele teme; accentul-ca-text pe alb foloseste varianta inchisa (`#1F66C9`).
|
||||
- [ ] Comutatorul light/dark existent (PRD 5.3) + anti-FOUC functioneaza in continuare.
|
||||
- [ ] `python3 -m pytest tests/test_web_tema_culori.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — accent azur + IBM Plex in light si dark, pill-uri rotunjite ca pe romfast.ro, contrast verificat.
|
||||
|
||||
### US-014: Selector de tema ciclic (Light/Dark/Petrol/Auto) (UI)
|
||||
**Ca** utilizator **vreau** ca butonul de tema sa cicleze prin mai multe teme (inclusiv una petrol)
|
||||
**pentru ca** vreau sa-mi aleg aspectul, nu doar light/dark, ca pe demoanaf.ro.
|
||||
|
||||
> Spec complet: `DESIGN.md` (sectiunile „Selector de tema" + paleta „Petrol").
|
||||
|
||||
- **Depinde de**: US-013 (paleta azur + variabilele de baza)
|
||||
- **Fisiere**: `app/web/templates/base.html` (script anti-FOUC + `[data-theme="petrol"]` + JS comutator + buton), `tests/test_web_selector_tema.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_selector_tema.py` — `test_petrol_theme_definit`, `test_buton_cicleaza_temele`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Butonul de tema **cicleaza** la click: Light → Dark → Petrol → Auto → Light; iconita + `aria-label`/`title` reflecta tema curenta.
|
||||
- [ ] **(must-fix, review A2/design)** Buton accesibil: `aria-label` anunta curenta + urmatoarea („Tema: Petrol, apasa pentru Auto"); schimbarea anuntata via regiune `aria-live="polite"`. Tooltip enumera ciclul. Cost de descoperire (4 teme intr-un buton fara optiuni vizibile) acceptat explicit in Non-Goals.
|
||||
- [ ] Tema noua **Petrol** definita ca `[data-theme="petrol"]` cu valorile din `DESIGN.md` (accent `--accent:#0E7C7B`, neutre petrol-inchise).
|
||||
- [ ] **Auto** urmeaza `prefers-color-scheme` (rezolva la Light/Dark azur); nu seteaza `data-theme` fix.
|
||||
- [ ] **(must-fix, review M4/design)** Scriptul anti-FOUC din `<head>` (`base.html:19-29`) cunoaste azi doar `light`/`dark`. Extinde-l atomic sa enumere toate cele 4 stari; o valoare `localStorage.theme` **legacy/necunoscuta** are fallback definit (nu blink, nu stare invalida). Rezolva „Auto" la light/dark inainte de primul paint.
|
||||
- [ ] **(must-fix, review S3/design)** Wordmark-ul „FAST" `#2E74D6` (albastru) coexista pe ecranul Petrol cu accentul teal `#0E7C7B` — verifica armonia/contrastul wordmark-ului (ROM rosu + FAST albastru) pe **toate cele 3 teme** concrete, nu doar light/dark.
|
||||
- [ ] Toate suprafetele raman lizibile (AA pe text principal) in fiecare din cele 3 teme concrete.
|
||||
- [ ] `python3 -m pytest tests/test_web_selector_tema.py -q` trece.
|
||||
- **Verificare E2E**: browser pe `/` — click pe buton cicleaza Light→Dark→Petrol→Auto; refresh pastreaza tema; Petrol are accent teal, fara blink la load.
|
||||
|
||||
### US-011: Butoane icon salvare/stergere vizibile + evidentiere modificari nesalvate (UI)
|
||||
**Ca** operator **vreau** butoane mici cu icon de salvare/stergere mereu vizibile pe randurile de
|
||||
mapari, evidentiate cand am modificari nesalvate **pentru ca** acum trebuie sa intru intr-un meniu
|
||||
contextual (kebab) si nu imi dau seama ca trebuie sa apas „Salveaza".
|
||||
|
||||
- **Depinde de**: US-010
|
||||
- **Fisiere**: `app/web/templates/_mapari.html`, `app/web/templates/base.html` (CSS/JS mic pentru stare „dirty"), `tests/test_web_mapari_actiuni.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapari_actiuni.py` — `test_butoane_icon_vizibile_pe_rand`, `test_fara_kebab_menu`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Pe „Mapari operatii salvate" si „Reguli automate (text)", actiunile Salveaza/Sterge sunt **butoane mici cu icon mereu vizibile** pe rand, nu ascunse in meniu kebab.
|
||||
- [ ] **(must-fix, review A1 — decizie de taste, vezi poarta)** Icon-urile sunt **SVG/text stilizate ca icon** (reuse `.icon-btn`), NU emoji brute 💾/🗑: emoji-ul nu se recoloreaza pe teme/dirty-state si randeaza inconsistent intre OS-uri. Numele accesibil din `aria-label`; glifa decorativa `aria-hidden`.
|
||||
- [ ] Meniul kebab (`position:fixed`) e **eliminat**.
|
||||
- [ ] **(must-fix, review S4)** Cand utilizatorul schimba select-ul unui rand, butonul de salvare devine **evidentiat** concret (ex. fundal `--accent` + nu emoji) ca semnal de „modificari nesalvate"; in starea normala e discret. Starea „dirty" e efemera per-render (un swap `outerHTML` o reseteaza — fara persistenta asteptata).
|
||||
- [ ] Confirmarea la stergere (`hx-confirm`) se pastreaza.
|
||||
- [ ] **(must-fix, review A6)** Accesibil: butoanele au `aria-label` descriptiv; pe mobil regula 44px e scopata ca icon-urile sa ramana **icon-size** (nu full-width ca butoanele normale `min-height:44px;width:100%`), cu zona de atingere adecvata.
|
||||
- [ ] `python3 -m pytest tests/test_web_mapari_actiuni.py -q` trece.
|
||||
- **Verificare E2E**: browser pe pagina Mapari — schimba un cod RAR la o mapare salvata → butonul de salvare se evidentiaza; salveaza/sterge din icon-uri vizibile.
|
||||
|
||||
## 4. Riscuri
|
||||
|
||||
- **Filtrarea pe data (US-001)**: `data_prestatie` poate veni in formate diferite (ISO cu/ fara ora,
|
||||
`T` vs spatiu). Mitigare: normalizeaza la primele 10 caractere doar cand sunt o data ISO valida;
|
||||
pastreaza excluderea valorilor ne-ISO (comportament existent) + test parametrizat pe formate.
|
||||
- **Paginare + filtru in Python (US-004)**: filtrele vehicul/data se aplica post-SQL; numararea
|
||||
totalului trebuie sa respecte filtrul, nu doar `COUNT(*)` SQL. Mitigare: numara dupa filtrare la
|
||||
scara actuala (plafon perf deja notat in cod) si testeaza `total` cu filtru activ.
|
||||
- **Editare operatie RAR (US-006)**: schimbarea codului trebuie sa recalculeze payload + idempotency
|
||||
ca la corectia `needs_data` existenta, altfel risca chei divergente. Mitigare: refoloseste exact
|
||||
calea de corectie/mapare existenta, nu una noua.
|
||||
- **Mutare Mapari din tab (US-009)**: deep-link-uri vechi `?tab=mapari` pot exista. Mitigare:
|
||||
alias/redirect, test ca nu da 404.
|
||||
- **Regresie pe contexte de eroare (US-008)**: simplificarea trebuie scopata DOAR pe formularul de
|
||||
editare; lista compacta + detaliu read-only + API raman pe catalogul 3-niveluri (PRD 5.4).
|
||||
|
||||
## 5. Intrebari deschise
|
||||
|
||||
> Rezolvate cu utilizatorul inainte de executie (poarta de aprobare PRD). Toate clarificate prin
|
||||
> AskUserQuestion in sesiunea de planificare:
|
||||
|
||||
- [x] Pill filtre: **un pill per categorie** (Date incomplete / Lipsa cod / Eroare) cu numar, nu un singur pill combinat.
|
||||
- [x] Paginare: **pagini numerotate**, 25 randuri/pagina, total vizibil, pastreaza filtrele.
|
||||
- [x] Editare operatie RAR: **doar pe stari editabile** (`needs_data`/`needs_mapping`/`error`); read-only pe `sent`/`sending`/`queued`.
|
||||
- [x] Layout Mapari: **De rezolvat prima**, apoi salvate / reguli text / formate coloane; fara ajutor, fara empty-text.
|
||||
- [x] Butoane icon: **evidentiere la modificare** (dirty state) pe langa vizibilitatea permanenta.
|
||||
- [x] Branding + paleta (US-012/US-013): rezolvate prin `design-consultation` → `DESIGN.md`.
|
||||
Wordmark „by ROMFAST" sub titlu (ROM rosu `#D1342F` + FAST albastru `#2E74D6`); accent functional
|
||||
**azur ROMFAST** (`#2E74D6`/`#1F66C9`), consistent cu romfast.ro; font IBM Plex Sans + Mono.
|
||||
- [x] Selector de tema (US-014): **buton ciclic** ca demoanaf.ro, set Light/Dark/Petrol/Auto;
|
||||
petrolul (directia initiala) revine ca tema selectabila.
|
||||
|
||||
## 6. Valuri de executie (graful de dependente)
|
||||
|
||||
```
|
||||
Val 1 (paralel, fisiere disjuncte):
|
||||
[US-001] fix filtrare data (routes.py)
|
||||
[US-002] operatie service in view (payload_view.py)
|
||||
[US-003] pill-uri filtre (_coada.html / _status.html)
|
||||
[US-005] VIN sub nr (_submissions.html)
|
||||
[US-008] eroare simpla in editare (_trimitere_detaliu.html)
|
||||
[US-009] Mapari in meniu (base.html / dashboard.html)
|
||||
|
||||
Val 2 (deblocate de Val 1):
|
||||
[US-004] paginare ← US-001 (acelasi handler)
|
||||
[US-006] editare op RAR ← (independent, dar atinge _trimitere_detaliu.html → dupa US-008)
|
||||
[US-007] op service detaliu ← US-002
|
||||
[US-010] restructurare Mapari ← US-009
|
||||
|
||||
Val 3:
|
||||
[US-011] butoane icon + dirty state ← US-010
|
||||
|
||||
Branding/tema (dupa design-consultation, ating base.html → serializeaza intre ele):
|
||||
[US-013] paleta azur ROMFAST + IBM Plex ← DESIGN.md (DONE)
|
||||
[US-012] header „by ROMFAST" + titlu centrat ← (independent, dar atinge base.html → dupa US-013)
|
||||
[US-014] selector tema ciclic + Petrol/Auto ← US-013
|
||||
```
|
||||
|
||||
> Nota orchestrare: US-006, US-007, US-008 ating toate `_trimitere_detaliu.html` → serializeaza-le
|
||||
> pe acelasi fisier (nu in worktree-uri paralele). La fel US-010/US-011 pe `_mapari.html`, si
|
||||
> US-012/US-013/US-014 pe `base.html` (toate trei ating header/CSS/script — un singur autor secvential).
|
||||
|
||||
---
|
||||
|
||||
## Raport autoplan (review CEO + Design + Eng)
|
||||
|
||||
> Rulat 2026-06-25. **Codex indisponibil** (usage limit pe contul OpenAI pana 2026-07-18) →
|
||||
> toate fazele degradate la **subagent-only** (`[codex-unavailable]`, single-model). Voci
|
||||
> independente Claude pe fiecare faza (fara context prealabil). Constatarile tehnice critice
|
||||
> verificate direct in cod de lead inainte de aplicare.
|
||||
|
||||
### Consensus (subagent-only — o singura voce per dimensiune)
|
||||
|
||||
```
|
||||
CEO : Premise valide (bug US-001 confirmat real). Recomandare = split branding/tema.
|
||||
DESIGN : Tare pe backend; lacune in stari vizuale + a11y controale noi (emoji, ciclu tema, pill).
|
||||
ENG : 4 must-fix verificate in cod (C1 _is_iso_date prefix; H1 total post-filtru;
|
||||
H3 US-006 nu e reuse pur; M4 card 3n in afara ramurii editabil).
|
||||
```
|
||||
|
||||
### Cross-phase themes (semnalate de 2+ voci independent)
|
||||
- **US-006 e cel mai riscant story** — CEO (riscant, confirma calea) + Eng (H3: NU e reuse pur, cere logica noua + validare nomenclator + assert pe cheia idempotency). Aplicat in AC.
|
||||
- **A11y a controalelor noi** — Design (emoji, ciclu tema, pill fara semantica) e clasa de probleme pe care TDD pe substring Jinja o trece dar livreaza un experience rupt la tastatura/screen-reader. Aplicat (SVG/text, aria-pressed, aria-live).
|
||||
- **Triplu-encoding contoare problema** — Design (pills + „Necesita atentia ta" + badge Mapari). Aplicat: pills inlocuiesc link-urile.
|
||||
|
||||
### Decizii auto (principiile 6) — must-fix aplicate in AC
|
||||
|
||||
| # | Faza | Decizie | Clasificare | Principiu | Rationament |
|
||||
|---|------|---------|-------------|-----------|-------------|
|
||||
| 1 | Eng | US-001: `_iso_date_prefix` (garda+comparatie pe `[:10]`) | Mecanic | P1/P5 | Fara asta bug-ul ramane; o singura solutie corecta |
|
||||
| 2 | Eng | US-004: total ramificat (SQL count fara filtru Python / fetch-all+slice cu filtru) | Mecanic | P1 | Contradictie Fisiere↔AC; SQL COUNT gresit cu filtru Python |
|
||||
| 3 | Eng | US-004: clamp `page`, ascunde pager la `pages<=1`, poll pastreaza pagina | Mecanic | P1 | Edge-cases + regresie poll |
|
||||
| 4 | Eng | US-006: camp `cod_prestatie` + validare nomenclator + assert cheie idempotency (NU reuse pur) | Mecanic | P1/P5 | Verificat in `routes.py:1009-1012` |
|
||||
| 5 | Eng | US-002/007: cheie payload distincta + conventie goala (`—` vs None) | Mecanic | P5 | Chei suprapuse azi; conventie ambigua intre stories |
|
||||
| 6 | Eng | US-008: muta cardul 3n in `{% if not editabil %}` + erori fara camp | Mecanic | P1 | Cardul e azi in afara ramurii editabil |
|
||||
| 7 | Design | US-003: pill-uri butoane reale (aria-pressed), inlocuiesc link-urile, matrice culori categorie (nu accent), sincron cu dropdown | Mecanic | P1 | A11y + anti triplu-encoding |
|
||||
| 8 | Design | US-009: scoate schela `role="tablist"` orfana + JS | Mecanic | P1 | Un tab in tablist = violare ARIA |
|
||||
| 9 | Design | US-011: icon-uri SVG/text (nu emoji brute), dirty-state concret, 44px scopat | Taste→aplicat | P1/P5 | User a scris emoji; SVG satisface intentia + a11y + recolorare (vezi poarta) |
|
||||
| 10 | Design | US-012/013/014: env badge in grila; FOUT pe tabulare; legacy `localStorage.theme`; ciclu a11y; wordmark pe 3 teme | Mecanic | P1 | Stari vizuale lipsa |
|
||||
|
||||
### Decizii de taste / User Challenge — la poarta (NU auto-decise)
|
||||
- **UC-1 (CEO, single-voice)**: split branding/tema (US-012/013/014) intr-un PRD separat + taie Petrol & selectorul ciclic. **Default = PASTRAM** (user a ales explicit in sesiune: Petrol ca tema, ciclu ca demoanaf, IBM Plex). CEO ruleaza ca o singura voce (codex jos) — recomandare, nu decizie.
|
||||
- **UC-2 (CEO)**: paginarea (US-004) poate fi prematura la scara actuala (purge 90z). **Default = PASTRAM** (alegere user); cost low (1 query de validat numarul max randuri/cont inainte de build).
|
||||
- **T-1 (Design A1)**: emoji 💾/🗑 (scrise de user) → SVG/text icon. Aplicat ca SVG (satisface ambele), surfata la poarta fiindca user a scris emoji literal.
|
||||
|
||||
### Deferat (TODOS, nu in 5.10)
|
||||
- Hardening GET-uri de listare globale/neprotejate (CLAUDE.md, semnalat de CEO ca valoare mai mare) — separat, nu expandam acest PRD.
|
||||
- Validare empirica „>200 randuri/cont" pentru US-004 si „cerere reala >2 teme" pentru US-014.
|
||||
|
||||
### Verdict poarta (2026-06-25)
|
||||
- **UC-1 → PASTRAM tot in 5.10** (14 stories, functional + branding/tema). User a confirmat la poarta; CEO single-voice = recomandare, nu verdict. Riscul FOUC/FOUT mitigat prin AC-urile adaugate.
|
||||
- **UC-2 → PASTRAM paginarea** (US-004). Cost mic, specificata corect dupa review.
|
||||
- **T-1 → icon-uri SVG/text** (nu emoji brute), aplicat in US-011.
|
||||
- **APROBAT.** Toate must-fix-urile aplicate in AC. Gata de executie (TDD prin echipa, vezi valuri §6) sau `/ralph`.
|
||||
|
||||
---
|
||||
|
||||
## Raport VERIFY
|
||||
|
||||
> Completat de subagentul verificator (context curat, fara transcriptul executiei) — ROADMAP §5.6. 2026-06-25.
|
||||
|
||||
**Verdict: PASS** — toate cele 14 stories PASS, 0 FAIL. Regresie completa **892 passed / 1 skipped / 0 failed** (+49 teste fata de baseline 843; skipped = test `live` RAR, opt-in). Backend trimitere (worker/masina stari/idempotenta/mapping) + schema-send NEATINSE (Non-Goals respectate).
|
||||
|
||||
PASS per story cu dovezi verificate direct in cod:
|
||||
- US-001 PASS — `_iso_date_prefix` folosit ATAT pentru garda CAT si comparatie (`routes.py`), interval inclusiv, ne-ISO excluse.
|
||||
- US-002 PASS — chei distincte `op_service_cod`/`op_service_denumire`, conventie goala `""` (nu `—`).
|
||||
- US-003 PASS — pill-uri `<button>` cu `aria-pressed`; needs_mapping `--warn`, needs_data/error `--err`; dropdown status eliminat (fara desync); lista ID-uri eliminata.
|
||||
- US-004 PASS — total ramificat (SQL COUNT fara filtru Python / fetch-all+slice cu filtru); clamp page; poll pastreaza pagina (OOB `f-page`); pager ascuns la `pages<=1`.
|
||||
- US-005 PASS — VIN element block-level (`<div>`) sub nr, garda VIN lipsa.
|
||||
- US-006 PASS — logica noua in `post_corectie_trimitere` (citeste `cod_prestatie`, valideaza nomenclator, injecteaza in `content["prestatii"][0]` inainte de `resolve_prestatii`); cheia idempotency difera; read-only pe sent/sending/queued.
|
||||
- US-007 PASS — Operatie service distinct de Operatie RAR, absent cand lipseste, in editabil + read-only.
|
||||
- US-008 PASS — card 3n in `{% if not editabil %}`; erori fara camp = rezumat top-of-form in editare.
|
||||
- US-009 PASS — Mapari in `#cont-menu` cu badge; tab-bar + `role=tablist` + JS sageti eliminate; `?tab=mapari` nu da 404.
|
||||
- US-010 PASS — ordine sectiuni corecta; `ajutor-mapari` + empty-text eliminate.
|
||||
- US-011 PASS — butoane `.icon-btn` SVG (fara emoji); kebab eliminat; dirty-state pe `change` select; `hx-confirm` pastrat.
|
||||
- US-012 PASS — header grid 3col centrat; env badge in grila; wordmark `by ROMFAST` (ROM `#D1342F`, FAST `#2E74D6`) text, nu PNG.
|
||||
- US-013 PASS — accent `#2E74D6`/`#1F66C9`/`#0E7C7B`; `@font-face` IBM Plex Sans/Mono. **Rezerva fonturi woff2 placeholder REZOLVATA post-VERIFY**: cele 8 fisiere au fost inlocuite cu subseturile reale IBM Plex (fontsource `@fontsource/ibm-plex-*@5.0.8`, woff2 valide latin + latin-ext, 400/500/700 sans + 400 mono).
|
||||
- US-014 PASS — `[data-theme=petrol]` `#0E7C7B`; ciclu Light->Dark->Petrol->Auto; anti-FOUC extins la 4 stari + fallback legacy; `aria-label` curenta+urmatoarea + `aria-live`.
|
||||
|
||||
**Limitari documentate:**
|
||||
- E2E browser NEPROBAT in sandbox (fara browser interactiv, consistent cu 5.8/5.9). Recomandat la deploy: `./start.sh test both --send` + browser pe `http://localhost:8000/` pentru proba vizuala (pill-uri/click, paginare >25, VIN sub nr, ciclu tema fara blink, header centrat, Mapari din ☰).
|
||||
- Live RAR `--send` neprobat (UI pur; backend trimitere neatins — risc minim).
|
||||
|
||||
### CLOSE — `/code-review high` (2026-06-25)
|
||||
|
||||
1 finding material reparat TDD (US-006b): US-006 acoperea doar `needs_data`/`needs_mapping`, dar AC + intrebarea deschisa cer si `error`. Fix: constanta separata `_EDITABILE_OP=(needs_data,needs_mapping,error)` pentru selectul de cod RAR; pe `error` editarea codului trece prin `/repune` (re-queue), cu validare nomenclator, re-rezolvare, canonicalize + `build_key` (cheie idempotency noua), check coliziune (pre-UPDATE + `IntegrityError`), `error→queued` (rar_error NULL / retry_count 0). `_CORECTABILE` + `post_corectie_trimitere` neatinse (fara regresie US-010/US-011). 9/9 teste editare; regresie completa **896 passed / 1 skipped / 0 failed**.
|
||||
|
||||
Findings minore (debt acceptat, non-blocante): (a) comparatie data bruta `d_prefix > data_pana` — sigura prin UI (`<input type=date>` zero-padded), vulnerabila doar la URL fabricat; (b) dublu `load_nomenclator` cand nomenclatorul e legitim gol; (c) linkurile de paginare reafiseaza `vehicul` cu majuscule (cosmetic); (d) duplicare cale validare+inject+idempotency intre `post_corectie_trimitere` si `post_repune_trimitere` (candidat de extras intr-un helper).
|
||||
|
||||
**US-012b (decizie user post-review):** in header se foloseste LOGO-ul PNG real (`/static/romfast_logo.png`, `.brand-logo` ~28px) in loc de wordmark-ul text din US-012. Fundal transparent + culori proprii -> lizibil pe toate temele. `test_web_header_branding.py` actualizat sa verifice `<img romfast_logo.png>` + `alt`. DESIGN.md actualizat. Regresie ramane 896 passed.
|
||||
|
||||
**Stare finala: VERIFY PASS + fix code-review (US-006b) + logo header (US-012b) aplicate. Gata de commit (poarta umana).**
|
||||
567
docs/prd/prd-5.11-ux-import-compact-preview-navigatie.md
Normal file
567
docs/prd/prd-5.11-ux-import-compact-preview-navigatie.md
Normal file
@@ -0,0 +1,567 @@
|
||||
<!-- /autoplan restore point: /home/claude/.gstack/projects/romfast-rar-autopass/main-autoplan-restore-20260626-095534.md -->
|
||||
# PRD 5.11 — Import compact + preview in format Trimiteri + navigatie + simplificare auto_send
|
||||
|
||||
**Stare**: inchis (2026-06-26 — verify-pass + code-review; vezi RAPORT VERIFY si RAPORT AUTOPLAN)
|
||||
|
||||
> Proces complet: `docs/ROADMAP.md` §5. Contractul RAR (sursa de adevar de contract):
|
||||
> `docs/api-rar-contract.md`. Starea trece: `draft → aprobat → in-executie → verify-pass → inchis`
|
||||
> (actualizata de lead). Acest PRD nu repeta strategia/contractul — le linkeaza.
|
||||
|
||||
## 1. Obiectiv
|
||||
|
||||
Reparam un set de bug-uri si frictiuni UX descoperite la dogfooding pe o baza goala (prima
|
||||
utilizare), confirmate in browser (E2E Playwright pe `prezentari_test.csv`):
|
||||
|
||||
1. **Dupa primul import ramai blocat** — wizard-ul se reseteaza la pasul 1, mesajul „vezi mai jos in
|
||||
Trimiterile tale" trimite la o sectiune care **nu exista in DOM** la first-run (`trimiteri-section`
|
||||
absent), iar lista nu apare fara reload manual.
|
||||
2. **Navigatie infundata** — din `?tab=mapari` / `?tab=jurnal` (si Cont/Integrare/Nomenclator) nu
|
||||
exista niciun link inapoi la Trimiteri; logo-ul nu e link; meniul hamburger nu are „Trimiteri".
|
||||
3. **Tabelul de preview (pasul 3) e neingrijit** — overflow orizontal taie coloanele Verificat?/Actiuni
|
||||
(butonul Editeaza), formularul de editare inline se taie, coloana „Note" scapa un **repr Python brut**
|
||||
(`[{'cod_op_service': ...}]`) iar „Stare" arata coduri brute (`needs_mapping`/`needs_data`).
|
||||
4. **`auto_send=0` produce o stare falsa** — un rand mapat (ex. OE-2) apare ca `needs_mapping` cu mesaj
|
||||
tehnic („cod mapat cu auto_send=0…") si fara actiune; bifa „In coada automat" adauga complexitate si
|
||||
confuzie. **Decizie utilizator (2026-06-26): scoatem complet conceptul auto_send din UI.**
|
||||
5. **Randul de filtre arata neuniform** — pill-urile au stil outline diferit de butonul solid Filtreaza
|
||||
si stau in dreapta lui; la hover deveneau rosu plin si textul devenea ilizibil.
|
||||
6. **Wizard-ul de import ocupa tot ecranul** — stepper + upload sunt doua carduri mari stivuite.
|
||||
|
||||
Toate sunt **UI/UX + o simplificare de mapare** (auto_send). Backendul de trimitere (worker, masina de
|
||||
stari de trimitere, idempotenta, contract RAR) ramane neatins, cu o singura exceptie controlata:
|
||||
logica de mapare nu mai „tine" randuri pe `auto_send` (US-001).
|
||||
|
||||
## 2. Non-Goals (anti scope-creep)
|
||||
|
||||
- **Nu** atingem worker-ul, reconcilierea, idempotenta sau contractul RAR.
|
||||
- **Nu** schimbam canalul API (`POST /v1/prezentari` / `/valideaza`) — doar UI web + `mapping.py`.
|
||||
(Campul `auto_send` din payload-ul API, daca exista, ramane acceptat dar ignorat — vezi US-001.)
|
||||
- **Nu** stergem coloanele DB `operations_mapping.auto_send` / `operation_text_rules.auto_send`
|
||||
(migrare distructiva inutila) — le lasam cu default `1` si nu le mai citim pentru a tine randuri.
|
||||
- **Nu** rescriem categoriile de filtrare cu etichetele din referinta (Facturate/Anulate/Diferente etc.)
|
||||
— pastram categoriile reale autopass (Toate / needs_mapping / needs_data / error). Adoptam doar
|
||||
**layout-ul si stilul** vizual din referinta (`image.png`).
|
||||
- **Nu** schimbam fluxul de erori pe 3 niveluri (5.4) — doar consumam helperele existente in preview.
|
||||
|
||||
## 3. Stories atomice
|
||||
|
||||
> Fiecare story: cea mai mica unitate care lasa sistemul functional. Backend + UI pentru acelasi
|
||||
> comportament = 2 stories. `Fisiere` + `Depinde de` complete (decid paralelizarea).
|
||||
|
||||
### US-001: Scoate „hold pe auto_send" din logica de mapare (backend)
|
||||
**Ca** operator **vreau** ca o operatie mapata la un cod RAR sa intre direct in coada **pentru ca**
|
||||
bifa auto_send introducea o stare `needs_mapping` falsa si confuza pe randuri deja corecte.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/mapping.py`, `app/web/routes.py` (DOUA puncte: `post_mapeaza_inline` ~1018-1069
|
||||
parsare form `auto_send` + delegare `reresolve_account`; ramura `has_no_auto_send` din
|
||||
`post_corectie_trimitere` ~1166 — referinta „~1160-1185" din draft arata gresit catre mapeaza),
|
||||
**`app/api/v1/import_router.py`** (importa `has_no_auto_send` ~linia 48, il foloseste in
|
||||
`_resolve_row_for_preview` ~233 — OMIS in draft; vezi AC „nu sterge simbolul"),
|
||||
`tests/test_mapping.py`, `tests/test_web_mapeaza.py`, **`tests/test_t6_auto_send.py` +
|
||||
`tests/test_text_rule_autosend.py` (de sters/rescris — encodeaza invariantul VECHI; vor deveni RED)**
|
||||
(~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_mapping.py` —
|
||||
`test_operatie_mapata_intra_in_queued_indiferent_de_autosend`,
|
||||
`test_regula_text_rezolvata_nu_mai_tine_randul`,
|
||||
`test_fara_stare_needs_mapping_pe_auto_send_oprit`,
|
||||
`test_niciun_rand_existent_nu_se_dezgheata` (AC dezghet),
|
||||
`test_canal_api_auto_send_ignorat_intra_queued` (`classify_prezentare`, `mapping_meta` auto_send=0 → `queued`)
|
||||
- **Acceptance criteria**:
|
||||
- [ ] `resolve_prestatii` NU mai marcheaza `regula_fara_autosend` / nu mai produce ramura
|
||||
`AUTO_SEND_OPRIT`; o operatie cu cod rezolvat (mapare exacta SAU regula text) → `queued`.
|
||||
- [ ] **`has_no_auto_send` NU se sterge — se neutralizeaza (`return False`) si ramane DEFINIT**
|
||||
(importat in `routes.py:70` SI `import_router.py:48`; stergerea simbolului → `ImportError` la
|
||||
load → app nu porneste). „Eliminat" si „intoarce mereu False" NU sunt echivalente: alege a doua.
|
||||
- [ ] Toate cele 4 callsite-uri tratate: clasificare (`mapping.py` ~413), `reresolve_account`
|
||||
(~664 + cheia stat `review_manual`), corectie (`routes.py` ~1166), preview (`import_router.py` ~233).
|
||||
- [ ] `/trimitere/{id}/mapeaza`: dupa mapare, randul cu cod valid trece `queued` (nu `needs_mapping`
|
||||
cu mesaj `auto_send`); raspuns fara cardul „auto-send oprit".
|
||||
- [ ] `needs_mapping` ramane DOAR pentru operatii fara cod RAR rezolvat (semantica reala).
|
||||
- [ ] Coloanele DB raman; `save_mapping` accepta inca `auto_send` (default 1) dar valoarea nu mai
|
||||
tine randuri (compat migrare). Niciun rand existent nu se „dezgheata" automat fara actiune.
|
||||
- [ ] **Randuri legacy `needs_mapping`-din-auto_send (cod deja prezent)** raman blocate pe test/prod
|
||||
fara afordanta de mapare (`_nemapate_pentru_submission` ~881 intoarce `[]`). DECIZIE explicita:
|
||||
requeue one-time la deploy (`reresolve_account` per cont, acum le trece `queued`) SAU documentam
|
||||
corectia inline ca singura iesire. (De notat in raportul VERIFY.)
|
||||
- **Verificare E2E**: `POST /trimitere/{id}/mapeaza` din panoul de detaliu (browser) → rand `queued`.
|
||||
|
||||
### US-002: Scoate bifa „In coada automat" din UI (Mapari + preview + detaliu)
|
||||
**Ca** operator **vreau** sa nu mai vad bifa auto_send nicaieri **pentru ca** nu mai are efect (US-001)
|
||||
si crea confuzie.
|
||||
|
||||
- **Depinde de**: US-001
|
||||
- **Fisiere**: `app/web/templates/_macros.html` (macro `autosend_toggle`),
|
||||
`app/web/templates/_mapari.html`, `app/web/templates/_preview_import.html`,
|
||||
`app/web/templates/_trimitere_detaliu.html`, `tests/test_web_mapari.py` (~5 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapari.py` —
|
||||
`test_mapari_fara_toggle_autosend`, `test_preview_panou_mapare_fara_autosend`,
|
||||
`test_detaliu_mapare_inline_fara_autosend`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Macro-ul `autosend_toggle` eliminat (sau golit) si scos din cele 3 sabloane.
|
||||
- [ ] Panoul „Operatii de mapat la cod RAR" (preview) si tab-ul Mapari salveaza maparea fara nicio
|
||||
referinta la „In coada automat" / „auto_send".
|
||||
- [ ] Coloana „In coada" din tabelul Mapari operatii salvate + reguli text dispare.
|
||||
- [ ] Niciun text rezidual „auto_send" / „Tine pentru verificare" in sabloane.
|
||||
- **Verificare E2E**: browser pe `/?tab=mapari` + panoul de mapare din preview — fara bifa.
|
||||
|
||||
### US-003: Preview pas 3 — format identic cu tabelul Trimiteri (UI)
|
||||
**Ca** operator **vreau** ca tabelul de preview sa arate exact ca lista Trimiteri **pentru ca** acum
|
||||
scapa text intern brut, taie coloane si formularul de editare.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_preview_import.html`, `app/web/templates/_preview_rand.html`,
|
||||
`app/web/templates/base.html` (refolosire `.tabel-trimiteri` + latimi `col-*` pt coloanele extra),
|
||||
**`app/web/payload_view.py`** (`prezentare_din_payload` accepta dict → `row.resolved` direct; OMIS),
|
||||
**stratul de adaptare in builderele de preview** (`app/web/routes.py` `_web_compute_preview` ~1851
|
||||
SI `app/api/v1/import_router.py` `_resolve_row_for_preview` ~122) — construieste un view-model tip
|
||||
`prez` + traduce `resolved_status`→eticheta, OMIS in draft,
|
||||
`app/web/labels.py` (NU reutilizare directa — vezi AC; necesita un map de stari PREVIEW),
|
||||
`tests/test_preview_import.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_preview_import.py` —
|
||||
`test_preview_nu_contine_repr_python` (fixture cu rand `needs_mapping`/unmapped REAL — altfel trece in gol),
|
||||
`test_preview_stare_eticheta_umana` (acopera `ok`/`needs_review`/`already_sent`/`duplicate_in_file`),
|
||||
`test_preview_foloseste_clasa_tabel_trimiteri`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **NU reutiliza `eticheta_stare`/`eticheta_scurta` direct: ridica `KeyError` pe starile de
|
||||
preview** (`ok`/`needs_review`/`already_sent`/`duplicate_in_file` — absente in `STARI_SUBMISSION`).
|
||||
Adauga un map de stari preview→eticheta umana (extinde `labels.py` sau map nou), inclusiv text
|
||||
pentru `already_sent`/`duplicate_in_file`.
|
||||
- [ ] **NU pasa `row.errors` (lista Python) in `motiv_uman`/`parse_erori`** (asteapta string JSON →
|
||||
`str()` + `json.loads` esueaza → fallback `raw[:160]` = ACELASI repr pe care US-003 il repara).
|
||||
Stratul de adaptare serializeaza erorile (`json.dumps`) sau formateaza uman inainte de render.
|
||||
- [ ] Tabelul de preview foloseste `.tabel-trimiteri`, fara overflow la 1280px (`scrollWidth <= clientWidth`).
|
||||
Necesita: `data-eticheta` pe TOATE celulele (carduri <768px le citesc via `td::before`) +
|
||||
latimi `col-*` pt cele 4 coloane extra (Verificat?/Actiuni/Note/etc.) sub `table-layout:fixed`.
|
||||
- [ ] Coloana „Note" → mesaj uman, **niciodata** repr Python (`[{'cod_op_service': ...}]`).
|
||||
- [ ] „Stare" → pill uman (ca la Trimiteri), nu cod brut `needs_mapping`/`needs_data`.
|
||||
- [ ] Vehicul (nr + VIN sub), operatie (+ cod RAR sub) randate via view-model `prez` (din `payload_view`).
|
||||
- [ ] Formularul inline `colspan` e **full-width sub rand** dar IESE din grila `table-layout:fixed`
|
||||
(rand `display:block` sau editor in afara tabelului fix), cu Salveaza/Anuleaza mereu vizibile.
|
||||
- [ ] Stare „filtrat la zero" (filter JS din `_preview_import.html` ascunde toate randurile) afiseaza
|
||||
un mesaj, nu tabel gol mut; filtrul coexista cu cardurile `td{display:block}`.
|
||||
- **Verificare E2E**: browser pasul 3 — coloane intregi, fara repr brut, editare necliata, mobil + desktop.
|
||||
|
||||
### US-004: Randul de filtre Trimiteri — layout + stil ca referinta (UI)
|
||||
**Ca** operator **vreau** un rand de filtre uniform si lizibil **pentru ca** pill-urile aveau alt stil
|
||||
decat butonul Filtreaza, stateau in dreapta si la hover deveneau rosu ilizibil.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_coada.html`, `app/web/templates/_pills.html`,
|
||||
`app/web/templates/base.html` (CSS `.pill-cat` + hover + layout), `tests/test_web_filtre.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_filtre.py` —
|
||||
`test_pill_uri_in_stanga_controalelor`, `test_pill_categorie_stil_uniform`,
|
||||
`test_quick_pills_data_seteaza_interval`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Layout ca `image.png`: pill-uri rapide de data (Azi / 7 zile / 30 zile / Custom) in STANGA,
|
||||
camp de cautare la mijloc, pill-uri de stare cu contoare la dreapta — un singur stil de pill.
|
||||
- [ ] Pill-urile NU mai stau `margin-left:auto` izolate la dreapta butonului.
|
||||
- [ ] **Stare activa, dezambiguizata**: pill-ul „Toate" (reset) = `--accent` plin; pill-urile de
|
||||
CATEGORIE = culoarea lor de categorie cand sunt active (NU toate accent — „accent plin" din
|
||||
draft contrazice schema per-categorie din `_pills.html`). Enumera token-ul de prim-plan per
|
||||
categorie ca sa garantezi AA, SAU foloseste o pereche fixa (alb pe saturat) cu valori verificate.
|
||||
- [ ] **Hover lizibil, regula explicita**: hover = `background:color-mix(in srgb, currentColor 12%,
|
||||
transparent)` (nu `filter:brightness` actual, nu rosu plin); pill-ul ACTIV suprima hover-ul.
|
||||
Acelasi mecanism pe reset si pe categorie.
|
||||
- [ ] **Focus pastrat**: inelul `:focus-visible` (outline accent) ramane pe toate cele 3 variante de
|
||||
pill dupa rescrierea CSS (US-004 atinge exact acest bloc — usor de pierdut).
|
||||
- [ ] Quick-pills de data seteaza `data_de`/`data_pana` (preset) si reincarca lista (HTMX), pastrand
|
||||
pill-ul de stare activ.
|
||||
- [ ] Functioneaza in Light/Dark/Petrol cu token-uri concrete enumerate (nu „verifica AA" abstract).
|
||||
- **Verificare E2E**: browser — comutare quick-pill data + stare, hover citibil pe toate temele.
|
||||
|
||||
### US-005: Link-uri de navigatie sub contoare pe toate paginile (UI)
|
||||
**Ca** operator **vreau** sa ajung oricand inapoi la Trimiteri / Mapari **pentru ca** din Mapari/Jurnal
|
||||
nu mai aveam cale de intoarcere.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/base.html` (header + `#cont-menu`), **AMBELE** `dashboard.html`
|
||||
SI `_status.html` (ambele randeaza `<div id="status-bar">` — pe cai diferite: full-page vs OOB/partial;
|
||||
„sau" din draft risca nav prezenta la load dar pierduta la refresh + duplicat de id),
|
||||
`tests/test_web_nav.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_nav.py` —
|
||||
`test_nav_trimiteri_mapari_pe_mapari`, `test_nav_trimiteri_pe_jurnal`, `test_logo_linkeaza_acasa`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Sub contoarele din `#status-bar`, pe FIECARE pagina (acasa/mapari/jurnal/cont/integrare/
|
||||
nomenclator): rand de link-uri `Trimiteri` + `Mapari` (cu badge needs_mapping daca exista),
|
||||
cu marcaj activ pe pagina curenta. **Sursa marcajului activ = o variabila de context numita
|
||||
explicit (ex. `tab_activ`, comparata cu `?tab=`), nu ghicita de implementator.**
|
||||
- [ ] Badge-ul `needs_mapping` reutilizeaza O SINGURA sursa (acelasi camp ca `_mapari_badge` din
|
||||
hamburger / `blocate_total`), nu un al doilea contor calculat diferit.
|
||||
- [ ] Logo-ul ROMFAST + titlul linkeaza la `/` (Trimiteri).
|
||||
- [ ] Meniul hamburger capata si „Trimiteri" (Acasa) ca prima intrare.
|
||||
- [ ] Niciun deadlock: din orice tab se ajunge la Trimiteri intr-un click.
|
||||
- **Verificare E2E**: browser — din Mapari si Jurnal, un click → Trimiteri.
|
||||
|
||||
### US-006: Import = container compact colapsabil (UI)
|
||||
**Ca** operator **vreau** ca importul sa nu ocupe tot ecranul **pentru ca** stepper-ul + upload-ul mare
|
||||
stau in cale dupa ce am deja trimiteri.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_acasa.html`, `app/web/templates/_upload.html`,
|
||||
`app/web/templates/_stepper.html`, `app/web/templates/base.html` (CSS/JS accordion),
|
||||
`tests/test_web_acasa.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_acasa.py` —
|
||||
`test_import_colapsat_cand_are_trimiteri`, `test_import_deschis_la_first_run`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Stepper + upload intr-UN singur container („Importa un fisier"), nu doua carduri.
|
||||
- [ ] Colapsat implicit cand contul are deja trimiteri (`are_trimiteri=True`); auto-deschis la
|
||||
first-run (`are_trimiteri=False`).
|
||||
- [ ] In timpul fluxului (mapcoloane/preview) containerul ramane deschis (nu se inchide intre pasi).
|
||||
- [ ] **Implementare cu `<details>` nativ** (disclosure CSS-only): serverul seteaza atributul `open`
|
||||
din `are_trimiteri`, deci „fara JS → degradare la deschis" si „colapsat la returning" sunt
|
||||
ambele corecte fara toggle JS (un toggle JS pur ar lasa returning-user-ul fara-JS cu ecranul
|
||||
permanent deschis = exact bug-ul reparat). `aria-expanded`/focus pastrate.
|
||||
- [ ] Ordinea de stivuire definita pe cele 4 combinatii (`are_creds` × `are_trimiteri`) — accordion,
|
||||
„Primii pasi", `trimiteri-section` nu se suprapun la first-run-cu-creds.
|
||||
- **Verificare E2E**: browser — first-run deschis; dupa ce exista trimiteri, colapsat; click → extins.
|
||||
|
||||
### US-007: Dupa commit, lista Trimiteri apare + se reimprospateaza automat (UI + glue)
|
||||
**Ca** operator **vreau** sa-mi vad trimiterile imediat dupa import **pentru ca** acum mesajul trimite la
|
||||
o sectiune inexistenta si nimic nu se actualizeaza fara reload.
|
||||
|
||||
- **Depinde de**: US-006
|
||||
- **Fisiere**: `app/web/routes.py` (`web_confirma_import` ~2757 intoarce AZI doar `_upload.html` via
|
||||
`hx-target=#import-section, outerHTML` — un swap `outerHTML` pe `#import-section` NU poate materializa
|
||||
un frate `#trimiteri-section`; alege mecanismul de dezvaluire — vezi AC), `app/web/templates/_upload.html`,
|
||||
`app/web/templates/_acasa.html`, `app/web/templates/_coada.html`, **`app/web/templates/_status.html` +
|
||||
`dashboard.html`** (OOB pe `#status-bar` — e in afara lui `acasa-section`, nu se actualizeaza la
|
||||
re-randarea `_acasa`/`_upload`), `tests/test_import_commit.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_import_commit.py` —
|
||||
`test_commit_raspuns_contine_trimiteri_section`,
|
||||
`test_commit_raspuns_seteaza_hx_trigger_trimiteriChanged` (asertie pe header literal),
|
||||
`test_first_run_dupa_commit_arata_lista`,
|
||||
`test_commit_actualizeaza_status_bar` (OOB contoare)
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **Mecanism de dezvaluire ales explicit**: fie (a) re-randam `_acasa.html` complet si retargetam
|
||||
pe `#acasa-section`, fie (b) emitem un OOB swap care injecteaza `_coada.html`. Minimal fata de
|
||||
codul actual (return `_upload.html`-only) = **(b)**. La first-run `#submissions-wrap` proaspat
|
||||
injectat isi declanseaza singur `hx-trigger="load"`.
|
||||
- [ ] **Raspunsul seteaza `HX-Trigger: trimiteriChanged`** (lipseste azi pe confirma; cf.
|
||||
`post_mapeaza_inline:1066`) — altfel la returning-user (sectiune deja in DOM) randurile noi
|
||||
`queued` nu apar. Atentie la dublu-load daca re-randam si `_acasa` (inofensiv, de notat).
|
||||
- [ ] Containerul de import se colapseaza, iar lista Trimiteri se reincarca automat si arata randurile.
|
||||
- [ ] **Contoarele `#status-bar` se actualizeaza via OOB swap** (sunt in `_status.html`/`dashboard.html`,
|
||||
in afara `acasa-section` — re-randarea `_acasa` nu le atinge), imediat, nu la poll-ul de 15s.
|
||||
- [ ] Mesajul de succes ramane onest („S-au pus in coada N prezentari") si pointeaza la o sectiune
|
||||
care chiar exista.
|
||||
- **Verificare E2E**: browser, baza goala — import → commit → lista apare + contoare actualizate, fara reload.
|
||||
|
||||
### US-008: Auto-refresh dupa actiuni, fara „Reincarca" manual (UI)
|
||||
**Ca** operator **vreau** ca lista sa se actualizeze singura dupa ce actionez **pentru ca** butonul
|
||||
manual „Reincarca" (nudge „Date noi") trecea neobservat (ex. dupa ce am mapat un cod si s-a trimis la RAR).
|
||||
|
||||
- **Depinde de**: US-007
|
||||
- **Fisiere**: `app/web/templates/base.html` (JS poller/nudge), `app/web/templates/_coada.html`,
|
||||
`app/web/templates/_submissions.html`, `tests/test_web_refresh.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_refresh.py` —
|
||||
`test_actiune_proprie_reincarca_automat`, `test_nudge_nu_mai_blocheaza_actualizarea`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Dupa o actiune proprie (mapare inline, corectie, repune, commit import) lista se reincarca
|
||||
automat (fara click pe „Reincarca").
|
||||
- [ ] Poll-ul de fundal: cand detecteaza date noi declansate de actiuni proprii, aplica refresh
|
||||
automat; nudge-ul manual ramane doar pentru schimbari externe (alt proces/worker) — sau se
|
||||
elimina daca devine redundant. (Decizie de implementare documentata in raport.)
|
||||
- [ ] Nu se reseteaza filtrul/pagina curenta la auto-refresh (pastreaza `#filtre-trimiteri`).
|
||||
- **Verificare E2E**: browser — mapare inline a unui cod → randul devine `queued`/`sent` fara click manual.
|
||||
|
||||
## 4. Riscuri
|
||||
|
||||
- **R1 — Scoaterea auto_send rastoarna „default-ul de siguranta CEO"** (reguli text / mapari tineau randul
|
||||
pentru verificare). Acceptat constient de utilizator (2026-06-26): mapped → queued direct. Mitigare:
|
||||
randurile genuin nemapate raman `needs_mapping` (nu pleaca), iar preview-ul are gate per-rand
|
||||
(„Verificat?") inainte de commit. Documentam in CLAUDE.md (invariantul auto_send) ca a fost retras.
|
||||
- **R2 — Preview in format Trimiteri**: tabelul Trimiteri presupune un view-model (vehicul/op/cod RAR);
|
||||
randurile de preview vin din alt drum (`_resolve_row_for_preview`). Risc de divergenta de campuri.
|
||||
Mitigare: refolosim `payload_view` / `labels` ca pe canalul Trimiteri; test anti-repr.
|
||||
- **R3 — Accordion + HTMX**: colapsarea nu trebuie sa ascunda pasii in timpul fluxului mapcoloane/preview.
|
||||
Mitigare: container deschis cat timp `import-section` randeaza un pas != upload slim; test dedicat.
|
||||
- **R4 — Regresie tabel**: `.tabel-trimiteri` are reguli responsive (carduri <768px) — preview-ul are
|
||||
coloane diferite (Verificat?/Actiuni). Mitigare: verificare E2E mobil + desktop.
|
||||
|
||||
## 5. Intrebari deschise
|
||||
|
||||
> Rezolvate cu utilizatorul la planificare (2026-06-26) — vezi raspunsurile incorporate mai sus:
|
||||
> Q1 filtre = **layout complet ca referinta** + fix hover lizibil; Q2 preview = **format ca Trimiteri**;
|
||||
> Q3 = **scoatem complet bifa auto_send** (Mapari + Trimiteri); Q4 = **acordeon compact + lista apare/
|
||||
> refresh automat dupa commit** + link-uri Trimiteri/Mapari sub contoare.
|
||||
|
||||
- (rezolvat) Pastram coloanele DB `auto_send`? → DA, default 1, ne-citite pentru hold (non-distructiv).
|
||||
- (deschis, minor) Nudge-ul „Date noi" pentru schimbari externe: il pastram redus sau il eliminam? →
|
||||
decizie la implementare (US-008), documentata in raport.
|
||||
|
||||
## 6. Valuri de executie (graful de dependente)
|
||||
|
||||
```
|
||||
Val 1 (paralel, fisiere disjuncte):
|
||||
[US-001] mapping.py + routes mapeaza (backend)
|
||||
[US-003] preview format Trimiteri (_preview_*.html, base.css)
|
||||
[US-004] filtre layout+stil+hover (_coada/_pills/base.css)
|
||||
[US-005] navigatie sub contoare (base.html/_status/dashboard)
|
||||
[US-006] import accordion compact (_acasa/_upload/_stepper)
|
||||
|
||||
Val 2 (deblocate de Val 1):
|
||||
[US-002] scoate toggle auto_send din UI (dep US-001; _macros/_mapari/_preview/_detaliu)
|
||||
[US-007] post-commit reveal+refresh (dep US-006; routes/_acasa/_upload/_coada)
|
||||
|
||||
Val 3:
|
||||
[US-008] auto-refresh dupa actiuni (dep US-007; base.js/_coada/_submissions)
|
||||
```
|
||||
|
||||
Atentie la fisiere fierbinti partajate intre stories (serializare de catre lead):
|
||||
`base.html` (US-004 CSS, US-005 header/menu, US-006 accordion JS) si `_coada.html` /
|
||||
`_preview_import.html` apar in mai multe stories — NU paralel pe acelasi fisier (vezi ROADMAP §5.5).
|
||||
|
||||
---
|
||||
|
||||
## Raport VERIFY
|
||||
|
||||
> Completat de subagentul verificator (context curat, 2026-06-26) in faza VERIFY — ROADMAP §5.6.
|
||||
> Executie: echipa de teammates Sonnet (lead orchestreaza, NU scrie cod), 6 runde TDD pe valuri cu
|
||||
> fisiere disjuncte; `base.html`/`routes.py`/`_coada.html`/`_status.html` serializate ca fisiere fierbinti.
|
||||
|
||||
**VERDICT GLOBAL: PASS → verify-pass / CLOSE.** 8/8 stories trec criteriile de acceptare, cu dovezi
|
||||
cod + teste + randare runtime (verificatorul a pornit aplicatia reala si a inspectat HTML-ul randat).
|
||||
|
||||
- **Suita**: `python3 -m pytest -q` → **929 passed, 1 skipped** (skip = `test_live_rar`, gated corect
|
||||
pe marker `live` + `AUTOPASS_LIVE_RAR`). Smoke boot `python3 -c "import app.main"` → OK (critic US-001:
|
||||
`has_no_auto_send` pastrat → fara ImportError). (Dupa adaugarea testelor cu nomenclatura PRD pe US-008:
|
||||
931 passed.)
|
||||
- **US-001** PASS — `has_no_auto_send`→`return False` (definit + importat in routes.py:72 si
|
||||
import_router.py:48); `resolve_prestatii` nu mai marcheaza `regula_fara_autosend`; ramura
|
||||
`AUTO_SEND_OPRIT` scoasa din classify; cele 4 callsite tratate; canal API auto_send=0 → `queued`;
|
||||
`needs_mapping` doar pt unmapped real; coloane DB raman; teste vechi rescrise (nu sterse). Decizie
|
||||
randuri legacy: `reresolve_account` le re-trece `queued` la deploy (documentat).
|
||||
- **US-002** PASS — macro `autosend_toggle` golit, simbol pastrat; **0** reziduuri auto_send /
|
||||
„In coada automat" in template-uri (grep) + 0 in dashboard-ul randat.
|
||||
- **US-003** PASS — `STARI_PREVIEW` map nou in `labels.py` (fara KeyError pe stari preview);
|
||||
`nota_umana_preview` (errors ca lista → fara repr brut); view-model `prez` din `payload_view` (accepta
|
||||
dict); `.tabel-trimiteri` + `col-*` + `data-eticheta` pe toate celulele + editor `tr.preview-edit`
|
||||
(scapa grila fixa) + mesaj „filtrat la zero".
|
||||
- **US-004** PASS — quick-pills data STANGA / cautare MIJLOC / pills stare DREAPTA; hover
|
||||
`color-mix(currentColor 12%)`, `:focus-visible` accent, categorie activa=`currentColor`, reset=`--accent`,
|
||||
activ suprima hover. (Nota minora: butonul „Custom" — vezi mai jos.)
|
||||
- **US-005** PASS — nav Trimiteri+Mapari in `_status.html` randat pe fiecare pagina (dashboard.html
|
||||
incarca fragmentul, fara duplicat de id), activ din `tab_activ`, badge din sursa unica
|
||||
`counts.needs_mapping`; logo+titlu link `/`; hamburger „Trimiteri" prima. Confirmat runtime
|
||||
(`aria-current="page"` pe Mapari la `?tab=mapari`).
|
||||
- **US-006** PASS — `<details id="import-details" {open if not are_trimiteri}>` nativ (fara toggle JS);
|
||||
runtime: first-run `open` + placeholder `#trimiteri-section`; returning colapsat + lista plina.
|
||||
- **US-007** PASS — `web_confirma_import` → `_upload` slim + OOB `_coada` (#trimiteri-section) + OOB
|
||||
`_status` (#status-bar) + header `HX-Trigger: trimiteriChanged`; `#submissions-wrap` se auto-incarca;
|
||||
mesaj de succes onest catre o sectiune care chiar exista.
|
||||
- **US-008** PASS — actiuni proprii emit `trimiteriChanged` → reincarcare; poller compara `data-v` si
|
||||
cheama `reincarcaTrimiteri()` cu `hx-include="#filtre-trimiteri"` (pastreaza filtru+pagina); nudge
|
||||
„Date noi" ELIMINAT (decizie documentata: distinctia propriu-vs-extern nu e posibila pe client).
|
||||
- **E2E**: PROBAT la nivel de aplicatie reala (uvicorn :8011, login sesiune, HTML randat real pentru
|
||||
accordion/nav/logo/filtre/zero-reziduu auto_send). NEPROBAT: click-through Playwright complet pe
|
||||
upload→preview→commit si testul live RAR `FINALIZATA` (gated, fara creds web in mediu) — acelasi
|
||||
status ca livrabilele anterioare; backend trimitere neatins, risc minim.
|
||||
- **Regresia de aur**: PASS la nivel de teste — `test_api` (POST /v1/prezentari) + `test_import_e2e`
|
||||
(import→commit→queued) + worker reconcile/rar_errors verzi; worker/idempotenta/`build_key`/contract RAR
|
||||
NEATINSE (non-goals respectate).
|
||||
|
||||
**Note minore (non-blocante, follow-up):**
|
||||
1. US-004 „Custom": AC enumera 4 quick-pills (Azi/7zile/30zile/**Custom**); doar 3 randate ca butoane
|
||||
(JS-ul suporta ramura `'custom'`, dar fara buton). De adaugat butonul SAU scos „Custom" din AC.
|
||||
2. Cod/comentarii vestigiale in `import_router.py` (`_motiv_clasificare` ramura `needs_mapping`/auto_send
|
||||
acum inaccesibila; comentarii care inca spun „auto_send gate"). Pur cosmetic.
|
||||
|
||||
---
|
||||
|
||||
<!-- ============================================================= -->
|
||||
<!-- /autoplan REVIEW REPORT — generat 2026-06-26, commit 412102b -->
|
||||
<!-- Codex indisponibil (usage limit) -> dual voices = subagent-only -->
|
||||
<!-- ============================================================= -->
|
||||
|
||||
# RAPORT AUTOPLAN
|
||||
|
||||
## Faza 1 — CEO (strategie & scope)
|
||||
|
||||
### 0A. Provocarea premiselor
|
||||
- **Premisa centrala (US-001): "auto_send hold = stare falsa".** Citind `mapping.py:252-254` si `434-447`, hold-ul era un **gate de siguranta intentionat**, nu un bug. Reformulare onesta: scoatem un gate de siguranta pentru ca frictiunea > riscul de ireversibilitate perceput. **GATE PREMISA: utilizatorul a confirmat (2026-06-26) scoaterea completa** — premisa acceptata constient, documentata in R1.
|
||||
- Premisa "preview-ul are gate Verificat? care compenseaza" e PARTIALA: gate-ul exista doar pe calea import web; canalul API + remaparea inline (`routes.py:1166`) trec direct in `queued`. Acceptat de utilizator la gate.
|
||||
- Restul premiselor (overflow tabel, repr Python in Note, hover rosu ilizibil, nav infundata, wizard mare, reveal post-commit rupt) = **confirmate in cod** (vezi Faza 3).
|
||||
|
||||
### 0B. Ce exista deja (mapare sub-probleme -> cod)
|
||||
| Sub-problema | Cod existent de refolosit |
|
||||
|---|---|
|
||||
| Etichete umane stare/motiv | `labels.py: eticheta_stare, motiv_uman, parse_erori` |
|
||||
| Viewmodel rand trimitere | `payload_view.py` + `r.prez.*` in `_submissions.html` |
|
||||
| Tabel responsive | `.tabel-trimiteri` + carduri <768px in `base.html` |
|
||||
| Pills stare + contoare | `_pills.html` + `filtreazaStare()` in `base.html` |
|
||||
| Refresh lista | HTMX `trimiteriChanged from:body` (deja cablat in `_coada.html`) |
|
||||
| Sectiune trimiteri | `#trimiteri-section` exista in `_coada.html` (doar conditionat de `are_trimiteri`) |
|
||||
|
||||
### 0C. Dream-state delta
|
||||
- ACUM: 8 frictiuni de first-run, una mascand o decizie de siguranta ireversibila.
|
||||
- ACEST PLAN: first-run curat (import -> preview ingrijit -> commit -> lista apare/refresh), nav fara fundaturi, mapare fara concept auto_send.
|
||||
- IDEAL 12 luni: un **E2E smoke de first-run** ca poarta de release (codifica scriptul de dogfooding) ca aceste 8 simptome sa nu reapara. **Recomandare deferata (vezi TODOS).**
|
||||
|
||||
### 0C-bis. Alternative (US-001)
|
||||
| Abordare | Efort | Risc | Verdict |
|
||||
|---|---|---|---|
|
||||
| Scoatere completa (ales) | mic | gate pierdut pe API/inline | **ALES de utilizator** |
|
||||
| Hold doar pe text-rules unattended | mic | minim (pastreaza net unde conteaza) | respins de utilizator la gate |
|
||||
| Default per-cont (ca `on_unmapped_error_default`) | mediu | reversibil, dar pastreaza complexitate | respins de utilizator |
|
||||
|
||||
### 0D-0F. Mod = SELECTIVE EXPANSION. Premisa confirmata la gate. Procedam.
|
||||
|
||||
### Voci duale CEO (subagent-only; Codex indisponibil)
|
||||
**CLAUDE SUBAGENT (CEO):** 11 constatari. Critice: (F1) US-001 e o decizie clasa-siguranta impachetata cu UX cosmetic — risc de aprobare-prin-asociere; (F6) regret 6 luni = o regula text gresita auto-trimite FINALIZATA terminal nerecuperabil. Hi: F3 (premisa laundering), F4 (mitigare R1 supraevaluata — nu acopera canalul API), F5 (cazul periculos real = text-rules unattended), F8 (alternativa default-per-cont neanalizata), F10 (over-bundling). Recomandare subagent: split US-001/002 din restul. **Utilizatorul a ales full removal la gate — F4/F5/F6 raman ca follow-up optional, nu blocant.**
|
||||
|
||||
```
|
||||
CEO DUAL VOICES — CONSENSUS:
|
||||
Dimensiune Claude Codex Consensus
|
||||
1. Premise valide? Partial N/A flag (gate trecut de user)
|
||||
2. Problema corecta? Da N/A confirmat de 1 voce
|
||||
3. Scope calibrat? Nu(F10) N/A flag (over-bundle, dar waves ok)
|
||||
4. Alternative explorate? Nu(F8) N/A flag (rezolvat la gate)
|
||||
5. Riscuri piata/legal acoperite? Partial N/A flag (R1 acceptat)
|
||||
6. Traiectorie 6 luni sanatoasa? Partial N/A flag (F6 -> TODOS)
|
||||
Codex = N/A (usage limit). Single-critical din 1 voce = flagat oricum.
|
||||
```
|
||||
|
||||
### NU in scope (confirmat)
|
||||
Worker / reconciliere / idempotenta / contract RAR / canal API (payload) / coloane DB auto_send (raman, default 1) / categorii filtrare (raman cele reale autopass) / fluxul de erori 3-niveluri.
|
||||
|
||||
### Registru Erori & Salvare (relevant acestui plan)
|
||||
| Suprafata | Esec | Salvare existenta |
|
||||
|---|---|---|
|
||||
| Import upload | fisier invalid / multi-foaie | `eroare_upload` / select foaie (`_upload.html`) |
|
||||
| Preview rand | cod necunoscut nomenclator | promovat la `cod_op_service` -> `needs_mapping` (nu se trimite raw) |
|
||||
| Commit | rand `needs_mapping` | exclus din commit (ramane in lista) |
|
||||
|
||||
### Sumar CEO
|
||||
Plan sanatos ca directie. Singurul flag strategic (US-001 fara control compensator pe calea unattended) **acceptat constient la gate**. US-003-008 = castiguri de first-run cu risc redus; US-007/008 cele mai sigure. Recomandare: pastram PRD-ul unit (waves deja serializeaza fisierele fierbinti), adaugam 1 item TODOS (E2E first-run + control optional text-rule).
|
||||
|
||||
## Faza 2 — Design (UI scope: DA)
|
||||
|
||||
### Voci duale Design (subagent-only; Codex indisponibil)
|
||||
**CLAUDE SUBAGENT (design):** constatari grounded in template-uri reale.
|
||||
- **F2.4 (CRITIC):** `eticheta_stare`/`eticheta_scurta` ridica `KeyError` pe starile de preview
|
||||
(`ok`/`needs_review`/`already_sent`/`duplicate_in_file`) — reutilizarea „ca la Trimiteri" crapa preview-ul. → US-003 AC.
|
||||
- **F3.1 (CRITIC):** „activa = accent plin" contrazice pill-urile colorate per-categorie (`_pills.html:10`).
|
||||
Reset = accent; categorie = culoarea ei. → US-004 AC.
|
||||
- **F2.1/F2.2/F2.3 (high):** celulele de preview n-au `data-eticheta` (carduri <768px ilizibile); coloanele
|
||||
extra n-au latimi `col-*` sub `table-layout:fixed`; editorul `colspan` mosteneste latimea care depaseste. → US-003 AC.
|
||||
- **F3.2/F3.3 (high):** token-uri de contrast/hover neenumerate per tema/categorie — cel mai mare risc de „haunt". → US-004 AC.
|
||||
- **F2.6 (high):** preview n-are view-model `prez`; adaptorul lipseste din Fisiere. → US-003 Fisiere.
|
||||
- **F4.1/F4.2 (high):** `#status-bar` definit in DOUA locuri; variabila de tab activ nenumita. → US-005 Fisiere+AC.
|
||||
- **F1.3 (medium):** degradarea fara-JS contrazice „colapsat la returning" — necesita `<details>` nativ. → US-006 AC.
|
||||
|
||||
```
|
||||
DESIGN LITMUS — CONSENSUS:
|
||||
Dimensiune Claude Codex Consensus
|
||||
1. Ierarhie info (accordion) ok* N/A confirmat (1 voce) *cu state-table
|
||||
2. Stari lipsa (preview) NU N/A flag CRITIC (F2.4) -> US-003
|
||||
3. Matrice pill hover/active/focus NU N/A flag CRITIC (F3.1) -> US-004
|
||||
4. Nav activ + badge Partial N/A flag (F4) -> US-005
|
||||
5. Specificitate AC Partial N/A flag (token-uri, view-model)
|
||||
Codex = N/A (usage limit).
|
||||
```
|
||||
|
||||
## Faza 3 — Eng (arhitectura, teste, risc)
|
||||
|
||||
### Diagrama de dependente (componente noi vs existente)
|
||||
```
|
||||
[import upload] -> _upload.html (#import-section, outerHTML swap)
|
||||
| US-006: invelit in <details> (open=are_trimiteri)
|
||||
v
|
||||
[web_confirma_import ~2757] --returneaza--> _upload.html (AZI: doar atat -> bug US-007)
|
||||
US-007 fix: + OOB inject _coada.html (#trimiteri-section) + HX-Trigger trimiteriChanged
|
||||
+ OOB #status-bar (_status.html / dashboard.html)
|
||||
|
|
||||
v
|
||||
[#submissions-wrap] --hx-trigger: load, trimiteriChanged from:body--> /_fragments/submissions
|
||||
| |
|
||||
v v
|
||||
_submissions.html (r.prez view-model) _pills.html (OOB) <- US-004 CSS
|
||||
|
||||
[mapping.py resolve_prestatii] --auto_send hold SCOS (US-001)--> status: queued | needs_mapping(real)
|
||||
has_no_auto_send: NEUTRALIZAT (return False), simbol PASTRAT (importat in routes.py + import_router.py)
|
||||
callsite-uri: clasificare ~413 | reresolve ~664 | corectie routes ~1166 | preview import_router ~233
|
||||
|
||||
[preview builders] _web_compute_preview ~1851 / _resolve_row_for_preview ~122
|
||||
US-003: + adaptor resolved_status -> eticheta umana + view-model prez (payload_view) + json.dumps(errors)
|
||||
```
|
||||
|
||||
### Voci duale Eng (subagent-only; Codex indisponibil)
|
||||
**CLAUDE SUBAGENT (eng):** 13 constatari verificate in cod.
|
||||
- **F1 (HIGH/boot-crash):** `import_router.py:48` importa `has_no_auto_send`, `:233` il foloseste — al 4-lea
|
||||
callsite, OMIS din US-001 Fisiere. Stergerea simbolului → `ImportError` → app nu porneste. „Eliminat" ≠ „return False". → US-001.
|
||||
- **F4 (HIGH):** `test_t6_auto_send.py` + `test_text_rule_autosend.py` encodeaza holdul vechi → devin RED, neprogramate. → US-001.
|
||||
- **F5/F6 (HIGH):** `motiv_uman`/`parse_erori` asteapta string JSON; `row.errors` (lista) → fallback `raw[:160]` = ACELASI repr. Necesita strat de adaptare in buildere. → US-003.
|
||||
- **F2 (medium):** referinta de linie „~1160-1185" arata gresit; sunt 2 puncte (mapeaza ~1032 form, corectie ~1166). → US-001.
|
||||
- **F3 (medium):** randuri legacy needs_mapping-din-auto_send raman fara afordanta (`_nemapate_pentru_submission` ~881 → `[]`). → US-001 AC dezghet.
|
||||
- **F7 (medium):** `payload_view.py` necesar dar omis din US-003 Fisiere. → US-003.
|
||||
- **F8/F9/F10 (HIGH):** US-007 — `outerHTML` pe `#import-section` nu materializeaza frate `#trimiteri-section`; lipseste `HX-Trigger`; `#status-bar` (alt fisier) nu se actualizeaza. → US-007.
|
||||
- **F11/F13 (low):** waves §6 supravand paralelismul pe `base.html` (3 stories, regiuni disjuncte → serializate de lead); `save_mapping` scrie auto_send=0 dupa US-002 (inofensiv cat e ignorat).
|
||||
|
||||
```
|
||||
ENG DUAL VOICES — CONSENSUS:
|
||||
Dimensiune Claude Codex Consensus
|
||||
1. Arhitectura sanatoasa? Da N/A confirmat (1 voce)
|
||||
2. Acoperire teste suficienta? NU N/A flag (F4 + gap-uri) -> test plan
|
||||
3. Riscuri performanta? ok N/A nimic flagat (batch lookup already_sent fara N+1)
|
||||
4. Securitate? ok N/A nimic nou (CSRF/auth neatinse de UI)
|
||||
5. Cai de eroare tratate? Partial N/A flag (boot-crash F1, KeyError F5/F6)
|
||||
6. Risc de deploy gestionabil? Partial N/A flag (randuri legacy F3)
|
||||
Codex = N/A (usage limit).
|
||||
```
|
||||
|
||||
### Test plan (artefact pe disc)
|
||||
`~/.gstack/projects/romfast-rar-autopass/main-5.11-test-plan-20260626.md` — 18 codepath-uri mapate la acoperire,
|
||||
suite de rulat (inclusiv smoke de boot `import app.main` dupa US-001), 3 gap-uri critice.
|
||||
|
||||
### Registru moduri de esec (cu flag-uri critice)
|
||||
| Mod de esec | Trigger | Gravitate | Mitigare in plan |
|
||||
|---|---|---|---|
|
||||
| App nu porneste | stergerea `has_no_auto_send` | CRITIC | US-001 AC: neutralizare, nu stergere + smoke boot |
|
||||
| Preview crapa la render | reutilizare `eticheta_stare` pe stari preview | CRITIC | US-003 AC: map de stari preview |
|
||||
| Repr Python reapare in Note | `row.errors` in `motiv_uman` | HIGH | US-003 AC: adaptor json.dumps |
|
||||
| Returning-user nu vede randuri noi | lipsa `HX-Trigger` pe confirma | HIGH | US-007 AC: header trimiteriChanged |
|
||||
| Randuri legacy blocate fara iesire | needs_mapping-din-auto_send pe prod | MEDIUM | US-001 AC: requeue one-time SAU doc corectie |
|
||||
|
||||
## Cross-phase themes
|
||||
- **Theme: vocabularul de stari preview ≠ submission** — flagat INDEPENDENT in Faza 2 (Design F2.4) si
|
||||
Faza 3 (Eng F5/F6). Semnal de incredere mare: US-003 nu poate reutiliza direct `labels.py`; necesita
|
||||
un strat de traducere + view-model. Cea mai importanta corectie a planului.
|
||||
- **Theme: Fisiere sub-dimensionate fata de AC** — US-001 (import_router), US-003 (payload_view + buildere),
|
||||
US-007 (_status.html) — toate omiteau fisiere pe care propriile AC le cer. Corectat in stories.
|
||||
|
||||
## Corectie graf de executie (§6)
|
||||
Wave 1 NU e complet paralel: `base.html` e atins de US-004 (CSS pill), US-005 (header/nav), US-006
|
||||
(`<details>`) — regiuni disjuncte, dar lead-ul le SERIALIZEAZA (3 edituri pe acelasi fisier). Paralelismul
|
||||
real al Wave 1 e mai mic decat anunta graful. `_coada.html` (US-004→007→008) e serializat oricum de dependente.
|
||||
|
||||
## Deferate la TODOS.md
|
||||
- E2E smoke de first-run ca poarta de release (CEO F2).
|
||||
- Control compensator optional pe auto-trimitere unattended (CEO F5/F6) — risc rezidual acceptat de user.
|
||||
|
||||
<!-- AUTONOMOUS DECISION LOG -->
|
||||
## Decision Audit Trail
|
||||
|
||||
| # | Faza | Decizie | Clasificare | Principiu | Rationament | Respins |
|
||||
|---|---|---|---|---|---|---|
|
||||
| 1 | CEO | Premisa auto_send: full removal | GATE (user) | — | Utilizatorul a confirmat la gate | hold text-rule-only; default per-cont |
|
||||
| 2 | CEO | Pastram PRD unit (nu split US-001/002) | Taste | P3/P6 | Waves serializeaza deja; pre-launch, risc real ~0 | split in 2 PRD-uri (subagent F1/F10) |
|
||||
| 3 | CEO | E2E first-run -> TODOS, nu in scope acum | Mechanical | P3 | Boil-lake separat, nu blocheaza 5.11 | inclus in 5.11 |
|
||||
| 4 | Eng | US-001: neutralizare has_no_auto_send, NU stergere | Mechanical | P5 | Stergerea = ImportError (import_router) = boot crash | stergere simbol |
|
||||
| 5 | Eng | US-001 Fisiere += import_router.py + teste vechi | Mechanical | P1/P2 | Callsite real + teste care devin RED | lasa Fisiere ca-n draft |
|
||||
| 6 | Eng | US-001: randuri legacy -> decizie requeue/doc | Taste | P1 | In blast radius; impact pe prod DB | ignora (lasa blocate tacit) |
|
||||
| 7 | Design+Eng | US-003: strat adaptare + map stari preview | Mechanical | P1 | Reutilizare directa crapa/repr — cross-phase theme | reutilizare directa labels |
|
||||
| 8 | Design+Eng | US-003 Fisiere += payload_view + buildere | Mechanical | P1/P2 | AC cere view-model prez inexistent pe calea preview | doar template+labels |
|
||||
| 9 | Design | US-003: data-eticheta + col-* + colspan escape | Mechanical | P1 | Altfel overflow/mobil rup (AC propriu) | doar clasa .tabel-trimiteri |
|
||||
| 10 | Design | US-004: reset=accent, categorie=culoare proprie | Mechanical | P5 | „accent plin" contrazice schema per-categorie | toate accent |
|
||||
| 11 | Design | US-004: token-uri hover/active/focus enumerate | Mechanical | P1/P5 | „verifica AA" abstract = haunt | lasa la implementator |
|
||||
| 12 | Design | US-005: AMBELE status-bar + var tab_activ | Mechanical | P5 | „sau" risca nav pierduta la refresh + dup id | un singur fisier |
|
||||
| 13 | Design | US-006: `<details>` nativ | Mechanical | P5 | Toggle JS pur rupe degradarea fara-JS | toggle JS |
|
||||
| 14 | Eng | US-007: OOB inject _coada + HX-Trigger + OOB status-bar | Mechanical | P1/P5 | outerHTML nu materializeaza frate; lipsa trigger | „include/dezvaluie" vag |
|
||||
|
||||
## Stare review
|
||||
- Faza 1 CEO: rulat (subagent-only). Premisa confirmata la gate.
|
||||
- Faza 2 Design: rulat (subagent-only). 2 critice -> AC corectate.
|
||||
- Faza 3 Eng: rulat (subagent-only). 1 boot-crash + cross-phase theme -> stories corectate.
|
||||
- Faza 3.5 DX: SARIT — fara suprafata developer-facing (planul exclude canalul API; UI = operator).
|
||||
- Codex: indisponibil toata sesiunea (usage limit pana Jul 18) -> toate vocile = subagent-only.
|
||||
598
docs/prd/prd-5.12-editare-modal-cont-obligatoriu-import.md
Normal file
598
docs/prd/prd-5.12-editare-modal-cont-obligatoriu-import.md
Normal file
@@ -0,0 +1,598 @@
|
||||
<!-- /autoplan restore point: /home/claude/.gstack/projects/romfast-rar-autopass/main-autoplan-restore-20260626-201417.md -->
|
||||
# PRD 5.12 — Editare unificata in modal + cont cu companie/email/CUI obligatorii + rafinari import (calendar data, mapare cu antet+prima inregistrare, un singur Salveaza, preview compact) + responsive tableta/mobil
|
||||
|
||||
**Stare**: inchis (verify-pass 2026-06-26; 8 stories TDD prin agent team, VERIFY context curat PASS + 1 FAIL remediat, /code-review high 3 buguri reparate; regresie 987 passed/1 skipped/0 failed; asteapta confirmare commit — poarta umana)
|
||||
|
||||
> Proces complet: `docs/ROADMAP.md` §5. Contractul RAR (sursa de adevar de contract):
|
||||
> `docs/api-rar-contract.md`. Starea trece: `draft → aprobat → in-executie → verify-pass → inchis`
|
||||
> (actualizata de lead). Acest PRD nu repeta strategia/contractul — le linkeaza.
|
||||
>
|
||||
> Continua 5.11 ([prd-5.11](prd-5.11-ux-import-compact-preview-navigatie.md)). **Backendul de
|
||||
> trimitere (worker, masina de stari de trimitere, idempotenta, contract RAR) ramane NEATINS.**
|
||||
> Atingeri de schema permise (ambele coloane noi, migrare defensiva `_migrate`, ca la 3.3b/3.5/3.6):
|
||||
> `accounts.email` (US-001) si `import_rows.reviewed` (US-007, marcaj „verificat" per rand de preview).
|
||||
> Vezi Non-Goals.
|
||||
|
||||
## 1. Obiectiv
|
||||
|
||||
Continuam dogfooding-ul de first-run inceput in 5.11. Sase frictiuni confirmate E2E in browser
|
||||
(Playwright pe `exemple/prezentari_test.csv`, 2026-06-26) — toate UI/UX, plus o regula de date pe
|
||||
conturi:
|
||||
|
||||
1. **Editarea unui rand din preview e rupta vizual si arunca o eroare JS.** Modul de editare inline
|
||||
(`tr.preview-edit` cu `display:block` intr-un tabel `table-layout:fixed`) colapseaza coloanele —
|
||||
antetul si formularul se randeaza pe verticala, caracter cu caracter (reprodus identic cu
|
||||
`image copy.png`). La click pe **Anuleaza** se arunca in consola
|
||||
`TypeError: Cannot read properties of null (reading 'htmx-internal-data')` (reprodus live).
|
||||
Decizie utilizator: **editarea trebuie sa fie un MODAL, ca la Trimiteri, refolosind ACELASI
|
||||
formular** (fara cod duplicat).
|
||||
2. **Data prestatiei se scrie doar manual** (input text cu hint `YYYY-MM-DD`). Trebuie sa se poata
|
||||
alege si din **calendar** (`<input type="date">` nativ, decizie utilizator — zero dependinte JS).
|
||||
3. **Conturile nu au reguli minime de identitate.** Confirmat in baza: toate conturile au `cui=NULL`,
|
||||
iar conturile create din CLI/teste nu au niciun utilizator → fara email. Decizie utilizator:
|
||||
un cont inregistrat trebuie sa aiba **obligatoriu companie, email si CUI**.
|
||||
4. **Maparea coloanelor nu arata datele.** Pasul 2 listeaza nume de coloana + 2 exemple stivuite, dar
|
||||
nu se vede clar **capul de tabel (numele coloanelor) + valorile primei inregistrari**, ca operatorul
|
||||
sa stie ce mapeaza.
|
||||
5. **Panoul „Operatii de mapat la cod RAR" cere un Salveaza per rand.** La un fisier cu N operatii
|
||||
nemapate sunt N butoane „Salveaza" si N submit-uri. Trebuie **un singur buton Salveaza** care
|
||||
salveaza toate maparile odata.
|
||||
6. **Tabelul de preview (pasul 3) nu e compact si are o coloana neclara.** Randurile sunt foarte inalte
|
||||
(VIN-ul se sparge pe verticala), iar coloana **„Verificat?"** nu are sens evident in acest pas
|
||||
(operatorul nu intelege bifa). Trebuie lista mai compacta si coloana clarificata/eliminata.
|
||||
7. **Pe tableta si mobil interfata arata prost si articolele din header se suprapun.** Header-ul are grila
|
||||
desktop `1fr auto 1fr` (`min-height:92px`, logo 60px) si un singur prag mobil `@media (max-width:767px)`,
|
||||
dar **nimic pentru tableta (768–1024px)** — acolo logo + titlu + badge mediu + comutator tema + versiune
|
||||
+ hamburger se inghesuie si se suprapun. Tot fluxul (header, import, preview, modal, Trimiteri, Mapari,
|
||||
Cont) trebuie **compact, functional si ergonomic** pe tableta si telefon, cu tinte touch si fara suprapuneri.
|
||||
|
||||
Toate sunt **UI/UX**, cu o singura exceptie de date controlata: identitatea contului (companie/email/CUI
|
||||
obligatorii, US-001/002).
|
||||
|
||||
## 2. Non-Goals (anti scope-creep)
|
||||
|
||||
- **Nu** atingem worker-ul, reconcilierea, idempotenta, `build_key`, masina de stari de trimitere sau
|
||||
contractul RAR.
|
||||
- **Nu** schimbam canalul API (`POST /v1/prezentari` / `/valideaza`) si nici logica de mapare
|
||||
(`mapping.py` `resolve_prestatii`). Maparea operatie→cod ramane neschimbata; doar UI-ul de mapare din
|
||||
pasul de import se reorganizeaza (US-005) si reuseaza `save_mapping`/`reresolve_account` existente.
|
||||
- **Nu** stergem coloanele DB `auto_send` (deja neutralizate in 5.11) si nu reintroducem conceptul.
|
||||
- **Nu** schimbam stocarea editarii de preview: ramane `import_rows.override_json` (Approach B din 3.6),
|
||||
ruta `POST /_import/{id}/rand/{i}/editeaza` ramane sursa de adevar; doar **suprafata** de editare trece
|
||||
din rand-inline in modal.
|
||||
- **Nu** facem editare in bloc / multi-rand si nici editare a operatiei/codului RAR din modalul de rand
|
||||
(codul se mapeaza din panoul „Operatii de mapat", ca azi).
|
||||
- **Nu** schimbam fluxul de login/parola; un cont poate avea in continuare mai multe loginuri (`users`),
|
||||
dar primeste un email canonic de contact pe `accounts` (US-001).
|
||||
- **Nu** rescriem validarea de continut (`validation.py`); `<input type="date">` produce tot `YYYY-MM-DD`,
|
||||
acceptat azi.
|
||||
|
||||
## 3. Stories atomice
|
||||
|
||||
> Fiecare story: cea mai mica unitate care lasa sistemul functional. Backend + UI pentru acelasi
|
||||
> comportament = 2 stories. Toate rutele web noi sub `require_login`, scoped pe contul din sesiune
|
||||
> (404 cross-account), CSRF pe toate POST-urile.
|
||||
>
|
||||
> **Cerinta transversala (toate story-urile cu UI): responsive obligatoriu.** Fiecare suprafata noua/atinsa
|
||||
> (US-002..007) se verifica E2E pe **3 viewport-uri: desktop (≥1280px), tableta (768–1024px) si mobil
|
||||
> (≤767px / ~390px)** — fara overflow orizontal (`scrollWidth <= clientWidth`), fara suprapuneri, tinte
|
||||
> touch ≥44px, modal full-screen pe mobil. US-008 acopera header-ul + cadrul global; fiecare story isi
|
||||
> verifica propria suprafata pe cele 3 viewport-uri.
|
||||
|
||||
### US-001: Backend — companie/email/CUI obligatorii pe cont (`accounts.email` + validari)
|
||||
**Ca** administrator al gateway-ului **vreau** ca orice cont sa aiba companie, email si CUI **pentru ca**
|
||||
azi conturile pot exista fara email (CLI/teste) si fara CUI, deci nu pot fi identificate fiscal/contactate.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/schema.sql` (coloana `accounts.email` + `_migrate` defensiv), `app/accounts.py`
|
||||
(`create_account` accepta+valideaza `email`; helper `account_is_complete`), `app/web/auth_routes.py`
|
||||
(signup: CUI devine obligatoriu; scrie `accounts.email`), `tools/account.py` (CLI create cere
|
||||
`--email` + `--cui`), `tests/test_accounts.py`, `tests/test_signup.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_accounts.py` —
|
||||
`test_create_account_fara_email_ridica`, `test_create_account_fara_cui_ridica`,
|
||||
`test_email_normalizat_lowercase_trim`, `test_migrare_adauga_coloana_email_idempotent`,
|
||||
`test_account_is_complete_false_pe_legacy_incomplet`;
|
||||
`tests/test_signup.py` — `test_signup_fara_cui_422`, `test_signup_scrie_email_pe_account`,
|
||||
`test_signup_cui_existent_mesaj_prietenos` (NU mesajul tehnic cu `activate --account`).
|
||||
- **Acceptance criteria**:
|
||||
- [x] Migrare: `accounts.email TEXT` (nullable la nivel de schema pentru conturile legacy), `_migrate`
|
||||
defensiv idempotent (ca `users.is_admin` la 3.3b). Contul de sistem id=1 ramane fara email (exceptat).
|
||||
- [x] `create_account(conn, name, cui, email, active)` — `name`/`cui`/`email` goale → `ValueError`
|
||||
cu cauza+fix (catalog `errors.py` daca exista cod potrivit); `email` normalizat (trim+lower);
|
||||
`cui` normalizat (trim+upper, ca azi). CUI duplicat → mesajul existent.
|
||||
- [x] Signup web: `cui` devine **obligatoriu** (azi optional); la succes scrie `accounts.email = email`-ul
|
||||
utilizatorului. Lipsa CUI → re-randare formular cu eroare (422), pastrand campurile.
|
||||
- [x] **CUI duplicat la signup = mesaj prietenos, NU cel tehnic** (decizie user 2026-06-26, optiunea 1):
|
||||
„Aceasta firma (CUI …) e deja inregistrata. Cere accesul de la administratorul contului." — fara
|
||||
referinta la CLI `activate --account`. **Model: 1 firma = 1 cont = 1 login**; fluxul de
|
||||
invitatie/alaturare a unui al doilea email pe aceeasi firma e deferit la TODOS (optiunea 2).
|
||||
- [x] **Canal de contact concret in mesaj** (T3 gate /autoplan, aprobat 2026-06-26): mesajul include un
|
||||
email/canal de suport configurabil din settings (ex. `support_email`); daca setarea lipseste,
|
||||
fallback la formularea de mai sus. Operatorul primeste un pas urmator real, nu doar „cere accesul".
|
||||
Nu mai lasam mesajul tehnic ridicat de `create_account` sa ajunga verbatim in signup — detectam
|
||||
CUI duplicat in handler-ul de signup si compunem mesajul prietenos acolo (NU `error=str(exc)`).
|
||||
- [x] CLI `tools/account.py create` cere `--email` + `--cui` (refuza fara ele); `--with-key` neschimbat.
|
||||
- [x] `account_is_complete(row)` (companie + email + CUI ne-goale) — helper pur, fara efecte.
|
||||
- [x] **NU** atinge `users`, `submissions`, worker-ul sau idempotenta.
|
||||
- **Verificare E2E**: TestClient — signup fara CUI → 422; signup complet → `accounts.email` populat;
|
||||
`create_account` fara email/cui → ValueError.
|
||||
|
||||
### US-002: UI — gate de activare + pagina Cont editeaza companie/email/CUI + banner legacy
|
||||
**Ca** operator/administrator **vreau** sa vad si sa completez companie/email/CUI **pentru ca**
|
||||
conturile incomplete (legacy) trebuie aduse la regula fara re-inregistrare.
|
||||
|
||||
- **Depinde de**: US-001
|
||||
- **Fisiere**: `app/web/templates/_cont.html` (sectiune noua „Date firma"), `app/web/routes.py`
|
||||
(ruta `POST /cont/date-firma` scoped sesiune + CSRF; context `account_meta`+`cont_incomplet`),
|
||||
`app/web/templates/admin.html` + `app/web/routes.py` (gate activare pe `account_is_complete`),
|
||||
`app/web/templates/_banner.html` sau `_acasa.html` (banner „Completeaza datele firmei"),
|
||||
`tests/test_web_cont.py`, `tests/test_admin.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_cont.py` —
|
||||
`test_cont_afiseaza_companie_email_cui`, `test_post_date_firma_actualizeaza`,
|
||||
`test_post_date_firma_cui_duplicat_eroare`, `test_banner_cont_incomplet_pe_legacy`;
|
||||
`tests/test_admin.py` — `test_activare_cont_incomplet_refuzata`.
|
||||
- **Acceptance criteria**:
|
||||
- [x] `_cont.html` are o sectiune „Date firma" (deasupra cheii API) cu companie + email + CUI editabile,
|
||||
prefilled din `accounts`; `POST /cont/date-firma` valideaza (reuse `create_account`-style) + CSRF +
|
||||
scoped sesiune; eroare pe CUI duplicat / camp gol, mesaj 3-niveluri.
|
||||
- [x] Banner ne-blocant „Completeaza datele firmei (email/CUI)" pe Acasa cand `account_is_complete` e fals;
|
||||
dispare dupa completare. NU blocheaza importul/uploadul.
|
||||
- [x] In panoul admin, butonul **Activeaza** e dezactivat (cu tooltip) pe conturi incomplete —
|
||||
nu activam la RAR un cont fara identitate completa.
|
||||
- [x] Fara regresie pe rutele existente din `_cont.html` (cheie API, creds RAR).
|
||||
- **Verificare E2E**: browser pe `/?tab=cont` — completez email+CUI → banner dispare; admin nu poate
|
||||
activa un cont incomplet.
|
||||
|
||||
### US-003: UI — pasul „Potriveste coloanele" arata antet + prima inregistrare
|
||||
**Ca** operator **vreau** sa vad numele coloanelor din fisier si valorile primului rand **pentru ca**
|
||||
sa stiu exact ce date mapez la fiecare camp RAR.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_mapcoloane.html`, `app/web/routes.py` (`web_upload_import` /
|
||||
`web_save_mapare_coloane` paseaza deja `sample_rows`; expune `prima_inregistrare`),
|
||||
`tests/test_web_mapcoloane.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapcoloane.py` —
|
||||
`test_mapcoloane_arata_cap_tabel_coloane`, `test_mapcoloane_arata_valori_prima_inregistrare`,
|
||||
`test_mapcoloane_fara_randuri_degradeaza` (fisier cu antet, fara randuri de date → fara crash).
|
||||
- **Acceptance criteria**:
|
||||
- [x] Deasupra (sau langa) randurile de mapare, un mic tabel orizontal cu **un cap de tabel = numele
|
||||
coloanelor din fisier** si **un rand = valorile primei inregistrari** (truncate la o lungime
|
||||
rezonabila, `title` pe valoare integrala). Foloseste `.tablewrap` pentru scroll orizontal pe mobil.
|
||||
- [x] Fiecare coloana din capul de tabel ramane vizual asociata cu select-ul ei de mapare (ex. aceeasi
|
||||
ordine, sau evidentiere la hover) — operatorul vede „coloana X (valoare „...") → campul canonic Y".
|
||||
- [x] Fisier fara randuri de date → se arata doar capul de tabel, fara „prima inregistrare" (fara crash).
|
||||
- [x] Nicio schimbare de backend de parsare/mapare; doar randare (datele exista deja in `sample_rows`).
|
||||
- **Verificare E2E**: browser pasul 2 — upload `prezentari_test.csv` → vad antetul real + valorile randului 1.
|
||||
|
||||
### US-004: UI+backend — un singur „Salveaza" pe „Operatii de mapat la cod RAR"
|
||||
**Ca** operator **vreau** sa salvez toate maparile de operatii dintr-un singur click **pentru ca**
|
||||
azi e cate un buton per operatie si trebuie apasat pe fiecare.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_preview_import.html` (panoul de mapare → un singur `<form>`),
|
||||
`app/web/routes.py` (ruta noua `POST /_import/{id}/mapare-operatii` plural; pastreaza
|
||||
`mapare-operatie` singular pentru compat sau o inlocuieste — vezi AC), `tests/test_web_mapare_op.py`
|
||||
(~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapare_op.py` —
|
||||
`test_mapare_operatii_salveaza_multiple_intr_un_post`,
|
||||
`test_mapare_operatii_ignora_randuri_neselectate` (op fara cod ales → nesalvata, nu eroare),
|
||||
`test_mapare_operatii_re_rezolva_blocatele` (randurile cu cod ales trec din `needs_mapping`).
|
||||
- **Acceptance criteria**:
|
||||
- [x] Panoul „Operatii de mapat la cod RAR" devine UN singur `<form>` cu un select per operatie +
|
||||
**un singur buton „Salveaza maparile"** la final.
|
||||
- [x] `POST /_import/{id}/mapare-operatii` primeste perechi `(cod_op_service, cod_prestatie)` (liste
|
||||
paralele), apeleaza `save_mapping` pentru fiecare operatie cu cod ales (reuse exact, fara logica
|
||||
noua de mapare), apoi **o singura** recompute `_web_compute_preview` + re-randare `#import-section`.
|
||||
- [x] Operatiile fara cod ales (`— alege cod RAR —`) sunt ignorate (nu produc eroare, nu se salveaza).
|
||||
- [x] Toggle-ul auto_send NU reapare (eliminat in 5.11).
|
||||
- [x] CSRF + scoped sesiune + guard batch committed (409) pastrate.
|
||||
- **Verificare E2E**: browser pasul 3 — aleg coduri pentru toate operatiile, un click pe „Salveaza
|
||||
maparile" → toate randurile trec din „Cod RAR lipsa", o singura re-randare.
|
||||
|
||||
### US-005: Refactor — formular de editare partajat (DRY) intre Trimiteri si preview
|
||||
**Ca** dezvoltator **vreau** un singur formular de editare de continut **pentru ca** sa nu existe cod
|
||||
duplicat intre modalul Trimiteri si editarea de preview (sursa bug-urilor inline din 3.6/5.11).
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/_form_editare.html` (NOU — partial cu campurile vehicul/data/odo),
|
||||
`app/web/templates/_trimitere_detaliu.html` (consuma partial-ul), `app/web/templates/_macros.html`
|
||||
(macro `camp` extins cu `tip='date'`), `tests/test_web_form_editare.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_form_editare.py` —
|
||||
`test_form_editare_are_input_date_pe_data_prestatie`,
|
||||
`test_trimitere_detaliu_foloseste_form_partajat`,
|
||||
`test_camp_macro_randeaza_type_date`.
|
||||
- **Acceptance criteria**:
|
||||
- [x] Partial `_form_editare.html` randeaza grila responsiva existenta
|
||||
(`repeat(auto-fit, minmax(200px,1fr))`) cu campurile: `nr_inmatriculare`, `vin`, `data_prestatie`,
|
||||
`odometru_final`, `odometru_initial`, plus map de erori per-camp (tipar `corectie_errors`).
|
||||
Parametrizat prin: URL de POST, valorile curente, harta de erori, eticheta butonului primar.
|
||||
- [x] **`data_prestatie` = `<input type="date">`** (calendar nativ); valoarea ramane `YYYY-MM-DD`.
|
||||
Daca valoarea curenta nu e `YYYY-MM-DD` valid, inputul degradeaza grijuliu (gol + hint), fara crash.
|
||||
- [x] `_trimitere_detaliu.html` randeaza acelasi partial in ramura `editabil` — comportamentul modalului
|
||||
Trimiteri (post `/corecteaza`, select cod RAR pe needs_data/needs_mapping) ramane identic.
|
||||
- [x] Macro `camp` suporta `tip='date'` fara sa strice apelurile `type='text'` existente.
|
||||
- **Verificare E2E**: browser — modalul Trimiteri (rand `needs_data`) arata un calendar la Data prestatie;
|
||||
salvarea+revalidarea functioneaza ca azi.
|
||||
|
||||
### US-006: UI — „Editeaza" din preview deschide MODALUL (acelasi formular), nu rand inline
|
||||
**Ca** operator **vreau** sa editez un rand de preview intr-un modal curat **pentru ca** editarea inline
|
||||
e rupta vizual si arunca eroare la Anuleaza.
|
||||
|
||||
- **Depinde de**: US-005
|
||||
- **Fisiere**: `app/web/templates/_preview_rand.html` (scoate ramura `editing`/`tr.preview-edit` +
|
||||
scriptul de mutual-exclusion; butonul „Editeaza" tinteste modalul global), `app/web/routes.py`
|
||||
(ruta GET fragment editare preview → randeaza `_form_editare.html` in `#detaliu-modal-body`;
|
||||
POST `/_import/{id}/rand/{i}/editeaza` ramane, dar raspunde cu inchidere modal + OOB pe rand+contoare),
|
||||
`app/web/templates/_preview_import.html` (foloseste modalul global `#detaliu-modal`),
|
||||
`tests/test_web_preview_edit.py` (~5 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_preview_edit.py` —
|
||||
`test_editeaza_preview_serveste_fragment_modal` (NU `tr.preview-edit`),
|
||||
`test_salvare_preview_inchide_modal_si_oob_rand`,
|
||||
`test_anuleaza_nu_lasa_rand_orfan` (regresie pe eroarea htmx null),
|
||||
`test_editare_preview_scoped_404_alt_cont`, `test_editare_batch_committed_409`.
|
||||
- **Acceptance criteria**:
|
||||
- [x] Butonul „Editeaza" pe rand face `hx-get` catre fragmentul de editare cu `hx-target="#detaliu-modal-body"`
|
||||
(acelasi mecanism de modal ca la Trimiteri, deschis prin clasa/markup existent in `base.html`).
|
||||
- [x] Fragmentul randeaza `_form_editare.html` cu POST la `/_import/{id}/rand/{i}/editeaza`,
|
||||
`hx-target="#detaliu-modal-body"`. La succes: **modalul se inchide** (`HX-Trigger: inchideModal`,
|
||||
ca la `/corecteaza`) si randul + contoarele se actualizeaza prin **OOB swap** (reuse `include_oob`).
|
||||
- [x] **Ramura `editing` / `tr.preview-edit` + scriptul inline de mutual-exclusion sunt ELIMINATE** din
|
||||
`_preview_rand.html` (sursa colapsarii pe verticala + a erorii `htmx-internal-data` la Anuleaza).
|
||||
- [x] „Anuleaza" = inchiderea modalului (mecanismul global), fara cerere catre `/_import/.../rand/{i}`,
|
||||
deci fara eroarea JS reprodusa. Test de regresie pe consola curata.
|
||||
- [x] Mutatie pura pe `override_json` pastrata (ruta neschimbata logic); scoping JOIN→404,
|
||||
guard committed→409 raman.
|
||||
- [x] Pe eroare de validare, modalul ramane deschis cu valorile + erorile per-camp (tipar Trimiteri).
|
||||
- **Verificare E2E**: browser pasul 3 — Editeaza → modal cu calendar + campuri; completez data → Salveaza →
|
||||
modal se inchide, randul trece pe „Gata de trimis", contoarele cresc; Anuleaza → modal se inchide,
|
||||
**0 erori in consola**.
|
||||
|
||||
### US-007: UI — preview compact + scoaterea coloanei „Verificat?"
|
||||
**Ca** operator **vreau** o lista de preview compacta si fara coloane neclare **pentru ca** randurile
|
||||
sunt prea inalte (VIN pe verticala) si nu inteleg bifa „Verificat?".
|
||||
|
||||
- **Depinde de**: US-006
|
||||
- **Fisiere**: `app/schema.sql` (coloana `import_rows.reviewed` + `_migrate` defensiv),
|
||||
`app/web/templates/_preview_rand.html`, `app/web/templates/_preview_import.html`
|
||||
(scoate coloana `col-verificat` + logica inline `reviewed_rows` din tabel),
|
||||
`app/web/templates/_form_editare.html` / fragmentul modal (buton „Confirma valorile" pe `needs_review`),
|
||||
`app/web/templates/base.html` (latimi `col-*` recalibrate, anti-overflow),
|
||||
`app/api/v1/import_router.py` + `app/web/routes.py` (citesc `reviewed` in `_resolve_row_for_preview` /
|
||||
`_web_compute_preview` ca `needs_review`-confirmat → `ok`; gate `n_confirmat` la commit foloseste
|
||||
`reviewed`, nu bife inline; ruta care seteaza `reviewed=1`), `tests/test_web_preview_compact.py`,
|
||||
`tests/test_import_review.py` (~7 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_preview_compact.py` —
|
||||
`test_preview_fara_coloana_verificat`,
|
||||
`test_preview_vin_nu_se_sparge_pe_verticala` (VIN intr-o singura linie / wrap controlat);
|
||||
`tests/test_import_review.py` —
|
||||
`test_needs_review_exclus_din_gata_pana_la_confirmare`,
|
||||
`test_confirmare_in_modal_seteaza_reviewed_si_devine_ok`,
|
||||
`test_reviewed_nu_intra_in_payload_sau_idempotency` (marcaj separat, NU camp de continut),
|
||||
`test_migrare_adauga_coloana_reviewed_idempotent`,
|
||||
`test_editare_valoare_pe_needs_review_reseteaza_reviewed` (daca schimbi valoarea, re-cere confirmare).
|
||||
- **Acceptance criteria**:
|
||||
- [x] Coloana **„Verificat?" eliminata** din tabelul de preview; antetul si celulele scad la 8 coloane.
|
||||
- [x] Randuri compacte: VIN nu se mai sparge pe verticala (latime minima pe coloana Vehicul / `white-space`
|
||||
controlat); fara overflow orizontal la 1280px (`scrollWidth <= clientWidth`); cardurile <768px raman.
|
||||
- [x] **Decizie inchisa (Q1): confirmare in modal, rand exclus pana confirmi.** Un rand `needs_review`:
|
||||
- apare cu pill „Verifica valori" + motivul concret in „Note" (data ambigua / formule Excel / coercion);
|
||||
- este **exclus din „gata de trimis"** (nu intra in `n_confirmat`) pana cand operatorul il deschide in
|
||||
modal (US-006) si apasa **„Confirma valorile"** (sau il corecteaza), ceea ce seteaza
|
||||
`import_rows.reviewed=1`; abia atunci randul devine `ok` la recalculul `_resolve_row_for_preview`.
|
||||
- [x] **Banner discoverability deasupra tabelului** (T1 gate /autoplan, aprobat 2026-06-26): cand exista randuri
|
||||
`needs_review`, un banner ne-blocant deasupra tabelului explica: „Randurile cu <pill>Verifica valori</pill>
|
||||
nu pleaca la RAR pana le deschizi si confirmi in modal." Fara el, gate-ul mutat din coloana vizibila in
|
||||
modal devine usor de ratat (operatorul crede ca pill-ul e informativ). Bannerul dispare cand
|
||||
`summary.needs_review == 0`.
|
||||
- [x] **Buton explicit „Confirma valorile"** (T2 gate /autoplan, aprobat 2026-06-26): in modal (US-006), randurile
|
||||
`needs_review` au un buton SEPARAT „Confirma valorile" care seteaza `reviewed=1` — atestare explicita,
|
||||
distincta de salvarea unei corectii de continut. NU se seteaza `reviewed=1` implicit la orice save
|
||||
(altfel operatorul ar atesta o valoare ambigua fara intentie). Salvarea unei CORECTII pe un rand deja
|
||||
confirmat reseteaza `reviewed` (vezi AC urmator).
|
||||
- [x] **Marcaj separat, nu camp de continut**: `import_rows.reviewed` (nullable/int, migrare defensiva) NU
|
||||
intra in payload, in `override_json` sau in cheia de idempotenta. Daca utilizatorul **schimba** o
|
||||
valoare a unui rand deja confirmat, `reviewed` se reseteaza (re-cere confirmare).
|
||||
- [x] Comitul ramane gate HARD pe `n_confirmat` (niciun rand ambiguu nu pleaca la RAR fara confirmare
|
||||
umana explicita) — acum derivat din `reviewed`, nu din bife inline `reviewed_rows`.
|
||||
- [x] Bara de confirmare („Trimite la RAR") si contoarele raman corecte dupa editari/confirmari (OOB),
|
||||
fara coloana Verificat?.
|
||||
- [x] Guard committed→409 si scoping JOIN→404 pe ruta de confirmare (acelasi tipar ca `/editeaza`).
|
||||
- **Verificare E2E**: browser pasul 3 (cu un xlsx cu data ambigua / VIN numeric) — lista compacta, fara
|
||||
coloana Verificat?, VIN pe o linie; randul `needs_review` ramane exclus din „gata de trimis" pana il
|
||||
confirm in modal („Confirma valorile") → devine „Gata de trimis", contorul creste.
|
||||
|
||||
### US-008: UI — responsive tableta + mobil (header fara suprapuneri + cadru compact/ergonomic)
|
||||
**Ca** operator pe telefon/tableta **vreau** o interfata compacta, fara articole de header suprapuse
|
||||
**pentru ca** azi pe mobil arata prost si elementele din header se calca unele pe altele.
|
||||
|
||||
- **Depinde de**: — (header/cadru global); coordonat cu US-006/007 pentru modal+preview pe mobil
|
||||
- **Fisiere**: `app/web/templates/base.html` (header grid + media queries tableta 768–1024 + mobil ≤767,
|
||||
modal full-screen, `.cont-menu`, tinte touch), eventual `app/web/templates/_status.html` /
|
||||
`_acasa.html` (contoare + nav pe randuri inguste), `tests/test_web_responsive.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_responsive.py` —
|
||||
`test_header_are_breakpoint_tableta` (exista reguli `@media` intre 768 si 1024 pentru header),
|
||||
`test_header_elemente_nu_au_min_height_fix_pe_mobil`,
|
||||
`test_modal_full_screen_pe_mobil` (clasa/regula prezenta). (Testele de markup/CSS; pixel-level la E2E.)
|
||||
- **Acceptance criteria**:
|
||||
- [x] **Header fara suprapuneri pe tableta (768–1024px)**: logo + titlu + badge mediu + comutator tema +
|
||||
versiune + hamburger se aseaza fara sa se calce (grid/flex care wrap-uieste sau ascunde versiunea/
|
||||
titlul lung); `min-height:92px` nu forteaza inghesuirea. Pe mobil (≤767px) raman regulile existente,
|
||||
verificate ca nu se suprapun la ~390px latime.
|
||||
- [x] **Compact + ergonomic**: spatieri reduse pe mobil, tinte interactive ≥44px (butoane, pill-uri,
|
||||
linkuri nav, intrari hamburger), fara dublu-scroll; modalul de editare (US-006) e **full-screen**
|
||||
pe mobil (nu o casuta minuscula).
|
||||
- [x] **Fara overflow orizontal** pe niciuna din paginile principale (Acasa/import, preview pas 3, Mapari,
|
||||
Cont, login/signup) la 768px si la ~390px (`scrollWidth <= clientWidth`).
|
||||
- [x] Contoarele de status + nav-ul „Trimiteri/Mapari" se aseaza pe randuri lizibile pe mobil (fara taiere).
|
||||
- [x] Light/Dark/Petrol/Auto raman corecte pe toate viewport-urile (fara regresie de tema).
|
||||
- **Verificare E2E**: browser Playwright cu `browser_resize` la **390×844 (mobil)**, **820×1180 (tableta)**
|
||||
si **1280×800 (desktop)** — screenshot pe Acasa/import, preview pas 3 (cu modal deschis) si Cont; header
|
||||
fara suprapuneri pe toate trei; 0 overflow orizontal; tinte touch ok.
|
||||
|
||||
## 4. Riscuri
|
||||
|
||||
- **R1 — Gate `needs_review` la scoaterea coloanei.** Coloana „Verificat?" era gate-ul HARD prin care
|
||||
randurile cu valori ambigue (data ambigua, formula Excel) intrau in trimitere doar dupa bifa umana.
|
||||
Scoaterea ei naiva ar auto-include randuri ambigue (declaratie ireversibila la RAR). Mitigare:
|
||||
confirmarea se muta in modalul de editare (US-007 AC); `n_confirmat` ramane gate HARD. Vezi Q1.
|
||||
- **R2 — Refactor formular partajat (US-005) atinge modalul Trimiteri (cale LIVE).** `_trimitere_detaliu.html`
|
||||
e folosit pentru corectii reale care re-trimit la RAR. Mitigare: US-005 = refactor fara schimbare de
|
||||
comportament; teste byte-compat pe post `/corecteaza` + regresia existenta verde inainte de US-006/007.
|
||||
- **R3 — `<input type="date">` si valori ne-`YYYY-MM-DD`.** Fisiere cu data in alt format ajung in editare
|
||||
ca text ne-valid pentru inputul date (s-ar goli). Mitigare: AC US-005 — degradare grijulie (gol + hint),
|
||||
fara pierdere tacuta; data ramane editabila si re-validata la salvare.
|
||||
- **R4 — Migrare `accounts.email`.** Conturi legacy raman cu `email=NULL`. Mitigare: coloana nullable +
|
||||
`account_is_complete` (banner + gate activare), nu hard-block; contul de sistem id=1 exceptat.
|
||||
- **R5 — Eroarea htmx `htmx-internal-data`.** Reprodusa la Anuleaza pe editarea inline. Mitigare: US-006
|
||||
elimina complet ramura inline + scriptul; test de regresie pe consola curata.
|
||||
- **R6 — Responsive = fisier fierbinte `base.html`.** US-008 atinge header + media queries, fisier partajat
|
||||
cu alte story-uri (US-007 latimi `col-*`). Mitigare: serializare la lead (NU paralel pe `base.html`);
|
||||
verificare pixel pe 3 viewport-uri ca breakpoint-ul de tableta nu strica desktop-ul/mobilul existent.
|
||||
|
||||
## 5. Intrebari deschise
|
||||
|
||||
> Se rezolva cu utilizatorul ÎNAINTE de executie (poarta de aprobare PRD).
|
||||
|
||||
- **Q1 (gate `needs_review`) — INCHIS (user, 2026-06-26): confirmare in modal, rand exclus pana confirmi.**
|
||||
Context — `needs_review` apare cand validarea TRECE dar parsarea fisierului a fost incerta, in 3 cazuri
|
||||
(sursa: `import_parse.py` + `import_router.py:201-230`), aproape exclusiv la **xlsx** (la CSV nu se
|
||||
declanseaza — de-aceea coloana e goala in cazul comun):
|
||||
1. **Data ambigua** — zi ≤12 si format neclar (`05.06` = 5 iun. sau 6 mai?).
|
||||
2. **Coloana cu formule Excel** fara valori calculate (rata mare de celule goale).
|
||||
3. **Coercion suspect** la citire xlsx — VIN numeric (pierde zerourile din fata) / odometru ca float.
|
||||
Decizie: scoatem coloana mereu-prezenta „Verificat?"; randul `needs_review` ramane **exclus din „gata de
|
||||
trimis"** pana e deschis in modal si **confirmat** („Confirma valorile") sau corectat, persistand
|
||||
`import_rows.reviewed=1` (marcaj separat, NU camp de continut → nu intra in payload/idempotenta).
|
||||
Implementat in US-007.
|
||||
- **Q2 (model cont-email) — INCHIS (user, 2026-06-26): model A** (email canonic pe `accounts`), cu
|
||||
**1 firma = 1 cont = 1 login**. CUI ramane unic; al doilea email pe acelasi CUI e respins la signup cu
|
||||
mesaj prietenos (US-001). Fluxul de invitatie/alaturare (mai multi utilizatori per firma) → TODOS.
|
||||
- **Q3 (CLI legacy `tools/account.py`)**: facem `--email`/`--cui` obligatorii rupe scripturile vechi de
|
||||
test? Daca da, pastram un flag `--allow-incomplete` doar pentru teste, sau actualizam fixture-urile.
|
||||
|
||||
## 6. Valuri de executie (graful de dependente)
|
||||
|
||||
```
|
||||
Val 1 (paralel, fisiere disjuncte):
|
||||
[US-001] accounts.email + validari companie/email/CUI (schema/accounts/auth_routes/cli)
|
||||
[US-003] mapcoloane: antet + prima inregistrare (_mapcoloane.html/routes)
|
||||
[US-004] un singur Salveaza pe operatii (_preview_import.html/routes)
|
||||
[US-005] formular de editare partajat (DRY) + input date (_form_editare/_trimitere_detaliu/_macros)
|
||||
|
||||
Val 2 (deblocate de Val 1):
|
||||
[US-002] Cont editeaza date firma + gate activare + banner (dep US-001; _cont/admin/routes)
|
||||
[US-006] Editeaza preview → modal (acelasi formular) (dep US-005; _preview_rand/_preview_import/routes)
|
||||
|
||||
Val 3 (deblocat de US-006; ating base.html → serializate):
|
||||
[US-007] preview compact + scoate „Verificat?" + gate review (dep US-006; _preview_*/base.css/import_router)
|
||||
[US-008] responsive tableta+mobil + header fara suprapuneri (base.html media queries; coordonat cu US-006/007)
|
||||
```
|
||||
|
||||
Fisiere fierbinti partajate (serializate de lead, NU paralel pe acelasi fisier): `routes.py`
|
||||
(US-001/002/003/004/006), `_preview_import.html` (US-004/006/007), `_preview_rand.html` (US-006/007),
|
||||
`base.html` (US-007 latimi `col-*` + US-008 header/media queries — serializate strict intre ele). Vezi ROADMAP §5.5.
|
||||
|
||||
---
|
||||
|
||||
## Raport VERIFY
|
||||
|
||||
> Faza VERIFY rulata de subagent verificator independent (context curat, PRD-only, ROADMAP §5.6),
|
||||
> 2026-06-26. Lead orchestrare prin agent team (8 teammates Sonnet TDD pe valuri cu fisiere disjuncte;
|
||||
> `routes.py` si `base.html` serializate ca fisiere fierbinti). Backend trimitere (worker, masina de
|
||||
> stari de trimitere, idempotenta `build_key`, contract RAR, canal API) NEATINS — confirmat
|
||||
> `git diff --stat` (app/worker/, app/idempotency.py, app/mapping.py, app/validation.py = 0 modificari).
|
||||
|
||||
### Rezultat: PASS (toate 8 stories)
|
||||
|
||||
- **Suita**: `python3 -m pytest -q` -> **987 passed, 1 skipped, 0 failed** (baseline 934 -> +53 teste noi).
|
||||
Live RAR `FINALIZATA` = opt-in indisponibil in mediu (normal, ca la livrabilele anterioare).
|
||||
- **PASS/FAIL per story** (dovezi cod + teste, verificator independent):
|
||||
- US-001 accounts email/CUI — PASS (migrare defensiva, create_account valideaza, account_is_complete
|
||||
id=1 exceptat, signup CUI obligatoriu + mesaj prietenos T3, CLI --email/--cui).
|
||||
- US-002 Cont date firma + gate activare + banner — PASS.
|
||||
- US-003 mapcoloane antet + prima inregistrare — PASS (confirmat E2E browser).
|
||||
- US-004 un singur Salveaza pe operatii — PASS (ruta plurala, D#12 skip invalid).
|
||||
- US-005 formular editare partajat + input date — PASS (D#5/D#6/D#10).
|
||||
- US-006 Editeaza preview -> MODAL — PASS (ramura inline eliminata, Anuleaza fara eroare htmx, E2E 0 erori consola).
|
||||
- US-007 preview compact + gate review in modal — PASS (reviewed marcaj separat, NU in payload/idempotenta;
|
||||
gate HARD pe ambele canale; T1 banner; T2 buton Confirma; D#9 reset).
|
||||
- US-008 responsive tableta + mobil — PASS (E2E pe 390/820/1280, header fara suprapuneri, D#13 verificat).
|
||||
- **Invariante critice**: R2 (submissions neatins dupa editare preview) PASS; reviewed in afara
|
||||
payload/override/idempotency PASS; migrari idempotente PASS; ramura inline `tr.preview-edit` eliminata PASS.
|
||||
|
||||
### VERIFY a gasit 1 FAIL -> remediat TDD, re-confirmat
|
||||
- FAIL: `signup.html` eticheta CUI „(optional)" + input fara `required` (contrazicea AC US-001 „CUI obligatoriu";
|
||||
serverul respingea corect 422 dar UI comunica gresit). Reparat TDD (eticheta `*` + `required`),
|
||||
test de lock `test_signup_html_cui_obligatoriu_ui`.
|
||||
|
||||
### Faza CLOSE — `/code-review high` (8 unghiuri prin subagenti, verificare cod first-hand)
|
||||
3 buguri reale reparate TDD (regresie finala 987 passed):
|
||||
1. **HIGH** — `confirma-review` folosea `hx-swap="none"` -> scriptul `updateN()` din continutul principal nu
|
||||
se executa -> `n_confirmat` ramanea stale -> „Trimite la RAR" pica pe gate HARD 422 (fluxul confirma->commit
|
||||
US-007 rupt la prima incercare). Fix: formularul Confirma valorile aliniat la `hx-target="#detaliu-modal-body"`
|
||||
`hx-swap="innerHTML"` (ca /editeaza).
|
||||
2. **MEDIUM** — email duplicat la signup arata mesajul gresit „firma e deja inregistrata" (`"deja folosit"`
|
||||
prindea si `ValueError("email deja folosit")` din `create_user`). Fix: detectie email-dup inaintea CUI-dup,
|
||||
mesaj specific emailului.
|
||||
3. **MEDIUM (a11y)** — butonul Editeaza din preview deschidea modalul ocolind `open()` (fara inert/focus-trap/
|
||||
focus-return). Fix: handler-ul global `htmx:beforeRequest` trateaza si `.btn-editeaza` -> `open()`; JS inline eliminat.
|
||||
Notat ca debt (neblocant): API preview re-deriva needs_review peste DB `resolved_status` cross-channel (web commit
|
||||
numara oricum `reviewed=1`); mesaje prietenoase „camp gol" dead-code in cont_date_firma/signup (edge mascat de HTML
|
||||
required); `zip()` truncheaza la liste POST inegale; `id` cont in mesajul CUI-duplicat; duplicari de cleanup
|
||||
(context modal, markup banner, N query nomenclator).
|
||||
|
||||
### Nedovedit in sesiune
|
||||
- Live RAR `FINALIZATA` prin `--send` (opt-in, lipsa creds/mediu) — risc minim, backend trimitere NEATINS.
|
||||
|
||||
---
|
||||
|
||||
## GSTACK REVIEW REPORT (/autoplan, 2026-06-26)
|
||||
|
||||
Branch: main · Commit: 283299f · Voci: Claude subagents (CEO/Design/Eng/DX) + verificare cod first-hand.
|
||||
**Codex = INDISPONIBIL** (usage limit, reset 2026-07-18) -> mod `[subagent-only]` pe toate fazele.
|
||||
Restore point: vezi comentariul HTML din capul fisierului. Test plan: `~/.gstack/projects/romfast-rar-autopass/main-prd5.12-test-plan-20260626.md`.
|
||||
|
||||
### Rezumat
|
||||
PRD matur: Q1/Q2 inchise de user, Non-Goals clare, graf de valuri, R1-R6. Rutele si fisierele citate exista
|
||||
toate in cod. Review-ul a confirmat fezabilitatea si a gasit **4 lacune de specificatie reale** (nu blocante,
|
||||
dar de inchis inainte de executie) + cateva rafinari. Niciun User Challenge (un singur model activ -> nu se
|
||||
poate forma consens cross-model; recomandarile de mai jos sunt sugestii, nu provocari).
|
||||
|
||||
### Decision Audit Trail
|
||||
|
||||
| # | Faza | Decizie | Clasificare | Principiu | Rationament | Respins |
|
||||
|---|------|---------|-------------|-----------|-------------|---------|
|
||||
| 1 | CEO | NU splitam in 5.12a/5.12b | Taste | P3/P6 | Valurile izoleaza deja US-001/002 (Val1/Val2) pe fisiere disjuncte; split adauga overhead de release fara castig tehnic pt. echipa mica | Split in 2 release-uri (CEO subagent) |
|
||||
| 2 | CEO | Respins „testeaza worker-ul intai" (F1/F3/F8) | Mechanical | P3 | Non-Goals ingheata explicit worker/contract/idempotenta; conflateaza acest PRD UI cu munca de backend separata | CEO subagent F1/F3/F8 |
|
||||
| 3 | CEO | First-run E2E smoke -> ramane in TODOS (deja listat) | Mechanical | P3 | Deja deferat din 5.11; recomandat, neblocant | A bloca 5.12 pe el |
|
||||
| 4 | CEO | needs_review: pastram gate-ul, nu cerem date de utilizare | Mechanical | P1 | Gate-ul e safety-critical (declaratie ireversibila RAR); US-007 muta UI-ul, nu sterge gate-ul | CEO F5 (gather usage data first) |
|
||||
| 5 | Eng | Partial partajat = DOAR campuri vehicul+data+err/fix; cod_prestatie select + nemapate_inline RAMAN in `_trimitere_detaliu` | Mechanical | P5/P4 | `_trimitere_detaliu` are 2 surse de cod (select + sectiune mapare inline) imposibil de absorbit fara branching fragil | Partial „atotcuprinzator" |
|
||||
| 6 | Eng | US-005 parametrizeaza si `fix_map` (+ aria-label cu VIN) | Mechanical | P1 | Forma preview are fix-hints + aria-label cu context VIN; lista PRD le omitea -> ar pierde info la extractie | A lasa lista PRD ca atare |
|
||||
| 7 | Eng | `import_rows.reviewed INTEGER DEFAULT 0` (nu NULL) | Mechanical | P5 | Gate-ul devine `reviewed=0` clar, fara ambiguitate NULL vs 0 | DEFAULT NULL |
|
||||
| 8 | Eng | Gate commit derivat din DB `reviewed` pe AMBELE canale; API `reviewed_rows` pastrat dar seteaza `reviewed=1` (contract stabil) | Mechanical | P1/P5 | Evita divergenta web/API si pastreaza contractul `/v1/import/.../commit` | A schimba doar web-ul (divergenta tacuta) |
|
||||
| 9 | Eng | reset `reviewed` la schimbare valoare se implementeaza in calea editeaza/override | Mechanical | P1 | E un AC US-007 fara loc de implementare numit; `apply_row_override` e locul | A-l lasa nespecificat |
|
||||
| 10 | Design | `<input type=date>` ne-ISO: gol + hint + valoare bruta in hidden, fara pierdere | Taste->auto | P1 | Previne pierderea tacuta de date pe formate Excel; backend deja marcheaza needs_review | A goli pur si simplu inputul |
|
||||
| 11 | Design | US-003 fisier fara randuri: mesaj explicit „antet fara randuri de date" + blocheaza Continua | Mechanical | P1 | Edge case altfel = esec tacut | Doar „fara crash" |
|
||||
| 12 | US-004 | Bulk mapping: validare per-item, skip invalid + sumar, restul salvate, 1 re-render | Mechanical | P1 | PRD acopera „fara cod = ignorat" dar nu „cod invalid pe 1 din N" | All-or-nothing |
|
||||
| 13 | US-008 | Modal full-screen mobil: VERIFICA, nu re-adauga (exista base.html:407) | Mechanical | P4 | Regula deja prezenta la `@media max-width:767px` | A re-implementa |
|
||||
| 14 | DX | Q3: actualizam fixture-urile via factory in `conftest.py`, FARA `--allow-incomplete` in prod | Taste | P5/P4 | Escape-hatch lasa o veruca in codul de productie; factory centralizat e curat si mai bun pe termen lung | `--allow-incomplete` flag |
|
||||
| T1 | Design | needs_review: banner persistent deasupra tabelului | Taste -> APROBAT user 2026-06-26 | P1/P5 | Gate-ul mutat in modal devine usor de ratat; bannerul il face explicit | Doar pill+tooltip |
|
||||
| T2 | Design/Eng | „Confirma valorile" = buton explicit separat (nu implicit pe save) | Taste -> APROBAT user 2026-06-26 | P5 | Atestare explicita pe valori ambigue; evita confirmarea accidentala | Implicit pe orice save |
|
||||
| T3 | DX | Mesaj CUI duplicat include canal de contact configurabil (fallback la actual) | Taste -> APROBAT user 2026-06-26 | P1 | Operatorul primeste un pas urmator real; detectie in handler signup, nu `str(exc)` | Pastreaza mesajul ca in PRD |
|
||||
|
||||
### NOT in scope (confirmat)
|
||||
- Worker, reconciliere, idempotenta, `build_key`, masina de stari de trimitere, contract RAR (Non-Goals).
|
||||
- Canal API `POST /v1/prezentari` / `/valideaza` si `mapping.resolve_prestatii` — neschimbate.
|
||||
- Multi-utilizatori per firma (flux invitatie/alaturare) -> TODOS.
|
||||
- First-run E2E smoke ca poarta de release -> TODOS (deja deferat din 5.11).
|
||||
- Split 5.12a/5.12b -> respins (vezi D#1).
|
||||
|
||||
### What already exists (de refolosit, nu reconstruit)
|
||||
- Modal global `#detaliu-modal` + `inchideModal` (`HX-Trigger-After-Settle`, routes.py:1235/1394) — US-006 il refoloseste.
|
||||
- `include_oob` pentru OOB swap rand+contoare — US-006/007 il refolosesc.
|
||||
- `save_mapping` / `reresolve_account` — US-004 le refoloseste (fara logica noua de mapare).
|
||||
- Macro `camp` exista INLINE in ambele forme (`_preview_rand.html:51`, `_trimitere_detaliu.html:98`) — US-005 il EXTRAGE (nu „extinde in `_macros.html`" cum spune lista de fisiere; `_macros.html` are azi doar `autosend_toggle` gol).
|
||||
- Modal full-screen mobil + tinte touch 44px — deja in base.html (`@media max-width:767px`, liniile 407-427). US-008 = tableta + verificare, nu rescriere.
|
||||
- `_migrate` defensiv idempotent (tipar `users.is_admin` 3.3b) — US-001/007 il urmeaza.
|
||||
|
||||
---
|
||||
|
||||
## Faza 1 — CEO (strategie & scope)
|
||||
|
||||
CEO DUAL VOICES — CONSENSUS:
|
||||
```
|
||||
Dimensiune Claude Codex Consens
|
||||
------------------------------------- -------- ------- ---------
|
||||
1. Premise valide? DA* N/A n/a (1 voce)
|
||||
2. Problema corecta? DA N/A n/a
|
||||
3. Calibrare scope? DISAGREE N/A -> taste (split?)
|
||||
4. Alternative explorate suficient? PARTIAL N/A n/a
|
||||
5. Riscuri competitive acoperite? DA N/A n/a
|
||||
6. Traiectorie 6 luni sanatoasa? DA N/A n/a
|
||||
* premise = decizii user deja luate (Q1/Q2 inchise, modal/calendar = „decizie utilizator")
|
||||
```
|
||||
**Examinat, nimic blocant pe strategie.** PRD-ul rezolva first-run friction confirmat E2E; scope-ul e calibrat
|
||||
prin valuri. CEO subagent a recomandat split-ul in 2 release-uri (D#1, respins) si a ridicat findings de „testeaza
|
||||
worker-ul" care cad in afara Non-Goals (D#2, respinse). Single-critical pastrat: Q3 (backward-compat CLI) — real,
|
||||
mutat la faza DX/Eng. Dream-state delta: 5.12 inchide first-run UX; ramane (separat) poarta E2E smoke + flux
|
||||
multi-user firma.
|
||||
|
||||
## Faza 2 — Design (UI/UX)
|
||||
|
||||
Litmus (Claude design; Codex n/a):
|
||||
```
|
||||
Dimensiune Scor Nota
|
||||
-------------------------------- ----- -------------------------------------------
|
||||
Ierarhie informatie (mapcoloane) 7/10 US-003 ok; recomandat cap-tabel sticky pe fisiere cu 15+ coloane
|
||||
Stari (load/empty/error/partial) 6/10 empty-file (US-003) si date ne-ISO (US-005) sub-specificate
|
||||
Gate needs_review in modal 6/10 LANDMINE: gate HARD mutat dintr-o coloana vizibila intr-un modal
|
||||
Responsive tableta (US-008) 7/10 breakpoint lipseste azi; spec sa fie pixel-exact, nu aspirational
|
||||
Interactiune modal (US-006/007) 6/10 „Confirma valorile" = buton separat vs implicit-pe-save (ambiguu)
|
||||
```
|
||||
**Issue-uri auto-decise:** D#10 (date ne-ISO), D#11 (empty-file mesaj). **Taste surfaced la gata:** banner
|
||||
discoverability pe needs_review (T1) + buton „Confirma valorile" explicit (T2).
|
||||
|
||||
## Faza 3 — Eng (arhitectura & teste)
|
||||
|
||||
ENG DUAL VOICES — CONSENSUS:
|
||||
```
|
||||
Dimensiune Claude Codex Consens
|
||||
--------------------------- -------- ------- ---------
|
||||
1. Arhitectura sanatoasa? DA(cond) N/A n/a — cond. pe partial corect parametrizat
|
||||
2. Acoperire teste? PARTIAL N/A n/a — vezi test plan, 4 gap-uri
|
||||
3. Riscuri performanta? DA N/A n/a — irelevant (UI/CRUD mic)
|
||||
4. Securitate? DA N/A n/a — CSRF+scoped pastrate pe rute noi
|
||||
5. Cai de eroare? PARTIAL N/A n/a — bulk mapping partial, date ne-ISO
|
||||
6. Risc deployment? DA N/A n/a — 2 migrari nullable defensive
|
||||
```
|
||||
|
||||
Diagrama arhitectura (componente noi vs existente):
|
||||
```
|
||||
US-001 create_account(+email) -> auth_routes.signup ──┐
|
||||
accounts.email (migrare) -> tools/account.py CLI ┤ (3 call-sites de actualizat)
|
||||
account_is_complete ┘
|
||||
US-005 _form_editare.html (NOU) <── _trimitere_detaliu.html (cod_prestatie select + nemapate_inline RAMAN aici)
|
||||
└──< _preview_rand.html (US-006: ramura inline ELIMINATA)
|
||||
US-006 preview „Editeaza" -> #detaliu-modal-body (GET fragment) -> POST /editeaza -> inchideModal + OOB
|
||||
US-007 import_rows.reviewed (migrare) -> _resolve_row_for_preview -> gate n_confirmat
|
||||
├── web routes.py /confirma (citeste DB reviewed)
|
||||
└── API import_router commit_import (reviewed_rows -> seteaza reviewed=1; contract stabil)
|
||||
US-008 base.html @media (768-1024) NOU + verificare 767 existent
|
||||
```
|
||||
|
||||
**Findings auto-decise:** D#5 (partial scope), D#6 (fix_map), D#7 (reviewed DEFAULT 0), D#8 (gate pe 2 canale),
|
||||
D#9 (reset reviewed in apply_row_override), D#12 (bulk partial). Test plan scris pe disc (44 codepath-uri,
|
||||
4 gap-uri marcate). **Invariant critic confirmat (R2):** editarea preview ramane override-only, NU re-queue —
|
||||
test obligatoriu (#24/#25 in test plan): dupa editare preview, `submissions` neatins.
|
||||
|
||||
## Faza 3.5 — DX (CLI + erori + contract API)
|
||||
|
||||
DX CONSENSUS (Claude; Codex n/a):
|
||||
```
|
||||
Dimensiune Nota
|
||||
--------------------------------- -------------------------------------------
|
||||
CLI create --email/--cui Q3 nerezolvat: a face flag-uri obligatorii rupe fixture-urile
|
||||
Mesaj eroare CUI duplicat prietenos da, dar „cere accesul de la admin" nu spune CUM
|
||||
Contract API commit reviewed_rows risc de divergenta tacuta -> rezolvat de D#8 + test #39
|
||||
Migrare fixture-uri recomandat factory in conftest.py (DX gap real)
|
||||
```
|
||||
**Auto-decis:** D#14 (Q3 -> factory, fara `--allow-incomplete`). **Taste surfaced:** mesaj CUI duplicat sa includa
|
||||
un canal de contact concret (T3).
|
||||
|
||||
### Cross-Phase Themes
|
||||
- **Tema A — Gate-ul needs_review (Design + Eng).** Design: mutarea in modal ascunde gate-ul (discoverability).
|
||||
Eng: gate-ul trebuie sa fie DB-backed pe ambele canale + reset la editare. Semnal high-confidence: tratati
|
||||
needs_review ca feature de sine statator in US-007, nu ca „stergere de coloana". -> T1 + D#8/D#9.
|
||||
- **Tema B — Sub-specificarea „Confirma valorile" (Design + Eng).** Ambii: cand se seteaza `reviewed=1`?
|
||||
Buton separat vs implicit pe save. -> T2.
|
||||
- **Tema C — Q3 backward-compat (CEO + DX + Eng).** Toate trei: a face email/CUI obligatorii rupe fixture-uri.
|
||||
-> D#14 (factory).
|
||||
|
||||
### Implementation Tasks (aggregate)
|
||||
_Niciun fisier `tasks-*.jsonl` per faza (autoplan ruleaza review-urile inline, nu skill-urile standalone)._
|
||||
Task-urile concrete = AC-urile din US-001..008 + cele 14 decizii din audit trail + 4 gap-urile din test plan.
|
||||
|
||||
### Status: DONE_WITH_CONCERNS
|
||||
Concerns (de inchis inainte de executie, niciunul blocant): cele 3 taste decisions de la gate (T1 banner,
|
||||
T2 buton confirma, T3 contact CUI) + integrarea celor 14 decizii in AC-urile US. Codex indisponibil -> review
|
||||
single-voice; re-ruleaza dupa 2026-07-18 daca vrei al doilea unghi adversarial.
|
||||
334
docs/prd/prd-5.13-responsive-compact.md
Normal file
334
docs/prd/prd-5.13-responsive-compact.md
Normal file
@@ -0,0 +1,334 @@
|
||||
<!-- /autoplan restore point: /home/claude/.gstack/projects/romfast-rar-autopass/feat-5.12-5.13-responsive-autoplan-restore-20260627-182914.md -->
|
||||
# PRD 5.13 — Responsive compact (mobil/tableta) + sistem de butoane + design.md
|
||||
|
||||
**Stare**: DRAFT — pentru /autoplan (implementarea NU e facuta inca)
|
||||
**Data**: 2026-06-27
|
||||
**Sursa de design**: [docs/design.md](../design.md) (sursa de adevar pentru planurile de design)
|
||||
|
||||
> Nota: o sesiune anterioara a explorat o implementare + mockup-uri si a fost REVENITA
|
||||
> (working tree readus la 5.12). Acest PRD ramane ca specificatie de planificat prin
|
||||
> `/autoplan`. Sectiunile „Livrabile" / „Raport VERIFY" de mai jos descriu directia
|
||||
> propusa si dovezile din explorare, NU stare livrata — re-validati prin autoplan.
|
||||
|
||||
## Context / problema
|
||||
|
||||
PRD 5.12 a marcat „responsive tableta/mobil" ca livrat, dar dogfooding-ul real
|
||||
(screenshot-uri `localhost_8010_.png`, `step3-preview.png`, `tablet-820.png`,
|
||||
`mobile-*.png`) a aratat ca paginile arata execrabil pe ecrane mici:
|
||||
|
||||
- **P0 break vertical**: in cardurile de Trimiteri pe mobil, eticheta lua `min-width:120px`
|
||||
fix iar valoarea (nod text intr-un flex) se strangea la ~0 si `word-break:break-word`
|
||||
o spargea **caracter cu caracter pe verticala** („B 0 7 5 8" pe coloana). Ilizibil.
|
||||
- **P0 stepper import**: 4 coloane egale cu text + `overflow:hidden` → pe tableta „Confirma
|
||||
trimiterea" era taiat; pe mobil 4 benzi minuscule cu text pe 3 randuri.
|
||||
- **P0 tabel preview pe tableta** (768–1024px): 8 coloane `table-layout:fixed` cu latimi
|
||||
fixe storceau vehicul+operatie → text rupt.
|
||||
- **P1 „afiseaza tot ca pe desktop"**: cardurile aratau toate cele 6–8 campuri, nu esentialul.
|
||||
- **P1 butoane exagerate**: `.tabel-card td button { width:100%; min-height:44px }` facea
|
||||
Salveaza + Sterge **doua blocuri full-width** unul sub altul; butoanele icon-only din
|
||||
„Mapari salvate" aveau **iconite invizibile** (SVG mic intr-un bloc colorat mare).
|
||||
- **P2**: mapare coloane cu scroll orizontal pe mobil; versiunea `vX.Y.Z` ocupa spatiu in
|
||||
header pe mobil; bara sticky de confirmare se rupea necontrolat.
|
||||
|
||||
Feedback user pe mockup-uri (2026-06-27):
|
||||
- Compactarea sa fie **si pe desktop** (ex. wizard-ul mai compact peste tot).
|
||||
- Pe **desktop** butoanele Salveaza/Sterge = **doar text** (fara iconita).
|
||||
- Pe **mobil** = iconite, dar un set modern, **recognoscibil** (Lucide stroke), nu cele vechi.
|
||||
- Nevoie de un **design.md** pe care planurile de design sa-l foloseasca.
|
||||
|
||||
## Decizii (confirmate cu user prin AskUserQuestion + mockup-uri)
|
||||
|
||||
1. **Directie**: carduri compacte, esential vizibil, butoane mici. (D: „da, dar ajustez")
|
||||
2. **Tableta (768–1024px)**: tabelele actionabile devin **carduri, 2 pe rand** (grid).
|
||||
3. **Scope**: pachet complet P0+P1+P2 + teste + acest PRD + ROADMAP + design.md.
|
||||
4. **Butoane**: desktop = text; mobil = iconita patrata 44px, set **Lucide stroke** (contur).
|
||||
5. **Wizard**: compact **peste tot** (inclusiv desktop): bara slim pe o linie; pe mobil
|
||||
„Pasul N din 4 · Titlu" + bara de progres.
|
||||
|
||||
## Livrabile
|
||||
|
||||
### design.md (nou)
|
||||
Sistemul de design: principii, tokeni (culoare 4 teme, tipografie IBM Plex, spatiere, radius,
|
||||
elevatie), breakpoints (768/1024), scara de control (44/36/32), componente (butoane, card,
|
||||
tabel→card, stepper, pill, formulare, modal), accesibilitate, si o sectiune „Pentru planurile
|
||||
de design". Sursa de adevar; planurile pornesc de aici.
|
||||
|
||||
### Cod (doar CSS + markup template; backend NEATINS)
|
||||
- **`_macros.html`**: macro `icon(name)` (Lucide save/trash/edit/plus, stroke) + `act_btn(label, ic, kind, attrs)`
|
||||
(buton de actiune responsiv: desktop text / mobil iconita).
|
||||
- **`base.html`** (CSS, inline):
|
||||
- Sistem de butoane `.btn-secondary/.btn-ghost/.btn-danger/.btn-sm` + default primar imbunatatit
|
||||
(font-weight 500, focus-visible).
|
||||
- Sistem `.act` / `.act-save` / `.act-del` / `.act-group`: desktop = text, mobil = iconita 44px.
|
||||
- Stepper compact `.stepper*` (track slim desktop/tableta; rezumat + bara progres mobil).
|
||||
- Card mobil Trimiteri/Preview **rescris**: stivuit compact, vehicul = titlu, stare = pill,
|
||||
`#`/checkbox ascunse, **fara gap fix de 120px** (fix break vertical).
|
||||
- **Tableta 768–1024px**: `.tabel-trimiteri` + `.tabel-card` → grid 2 carduri/rand.
|
||||
- Versiune ascunsa pe mobil; bara sticky confirmare compacta; mapare coloane stivuita full-width.
|
||||
- Coloana actiuni preview 92→104px + `.btn-editeaza { white-space:nowrap }`.
|
||||
- **`_stepper.html`**: rescris pe clasele compacte (fara stiluri inline inalte).
|
||||
- **`_mapari.html`**: butoanele icon-only inlocuite cu `act_btn` (salvate + reguli-text + formate-coloane).
|
||||
|
||||
### Teste
|
||||
- `test_web_responsive.py`: aserturile existente pastrate (toate trec).
|
||||
- `test_web_mapari_actiuni.py`: actualizat de la `.icon-btn` la sistemul `.act` (act-save/act-del,
|
||||
aria-label pe fiecare, `.act-ic` prezent), docstring marcat „superseda 5.10".
|
||||
|
||||
## Invariante respectate
|
||||
- Breakpoint unic 768px + densitate 1024px; un singur bloc `@media (max-width:767px) {` principal
|
||||
pe care se bazeaza testele (regulile noi adaugate inauntru, nu in blocuri noi inaintea lui).
|
||||
- Tabelele dense read-only (Jurnal/Nomenclator/Admin) raman scroll-contained, NU se cardifica.
|
||||
- Backend trimitere (worker, masina stari, idempotenta, contract RAR, canal API, mapping,
|
||||
validation) NEATINS. Zero schema. Pur CSS + markup.
|
||||
|
||||
## Raport VERIFY (live, app pornit cu DB seedata, Playwright 390/820/1280)
|
||||
- **Trimiteri mobil 390**: carduri compacte, pill stare + vehicul bold + operatie+cod, fara
|
||||
break vertical. Header compact, versiune ascunsa. PASS.
|
||||
- **Trimiteri tableta 820**: grid 2 carduri/rand, fara scroll orizontal. PASS.
|
||||
- **Trimiteri desktop 1280**: tabel complet neschimbat (fara regresie). PASS.
|
||||
- **Mapari mobil 390**: Salveaza = iconita discheta albastra, Sterge = iconita cos rosu (Lucide,
|
||||
patrate 44px, recognoscibile), NU blocuri full-width. PASS.
|
||||
- **Wizard import**: compact pe o linie pe desktop (✓ Incarca · ✓ Coloane · 3 Verifica · 4 Confirma)
|
||||
+ ajutor sub; pe mobil „Pasul 3 din 4 · Verifica" + bara progres. PASS.
|
||||
- **Preview import mobil 390**: carduri compacte per rand + bara confirmare compacta. PASS.
|
||||
- Regresie pytest: vezi ROADMAP (suita verde).
|
||||
|
||||
## Debt notat (neblocant)
|
||||
- Duplicarea pill stare + `eticheta_problema` pe error/needs_* arata redundant in carduri
|
||||
(„Eroare / Eroare") — logica de continut in `routes.py`/`labels.py`, nu responsive.
|
||||
- Filtrele de data (Azi/7zile/30zile/Custom) stivuiesc full-width pe mobil (4 randuri); ar putea
|
||||
fi grid 2x2 — imbunatatire viitoare.
|
||||
|
||||
---
|
||||
|
||||
<!-- ============================================================ -->
|
||||
<!-- GSTACK REVIEW REPORT — /autoplan (subagent-only; codex usage-limited) -->
|
||||
<!-- ============================================================ -->
|
||||
|
||||
# GSTACK REVIEW REPORT
|
||||
|
||||
> Pipeline: CEO -> Design -> Eng -> Final Gate (DX skipped: produs end-user, nu unealta de developer).
|
||||
> Voci: Claude subagent independent pe fiecare faza. **Codex INDISPONIBIL** (usage limit
|
||||
> OpenAI pana la 2026-07-18) -> toate fazele ruleaza `[subagent-only]`. Consensul nu poate
|
||||
> fi "CONFIRMED de ambele modele"; o singura voce.
|
||||
|
||||
## Faza 1 — CEO (strategie & scope)
|
||||
|
||||
### CEO dual voices
|
||||
- **Codex (CEO):** `[codex-unavailable: usage limit]`.
|
||||
- **Claude subagent (CEO):** rulat foreground, independent. Findings:
|
||||
- **F1 (high):** premisa "utilizarea pe mobil e reala" e asumata, nedovedita. Submiterea RAR
|
||||
e data-entry din xlsx/csv la receptia service-ului = task desktop/tastatura. Wizardul
|
||||
(upload -> mapare -> preview -> commit) e greu utilizabil pe telefon indiferent de CSS.
|
||||
Niciun semnal real (analytics/cerere user) citat; doar screenshot-uri ale propriului render urat.
|
||||
- **F2 (high):** reframe — durerea reala sunt bug-urile de layout de pe DESKTOP (break vertical,
|
||||
stepper taiat, butoane full-width, iconite invizibile), nu "responsive". Tier 1 = bug-uri +
|
||||
sistem butoane + design.md (valoare clara, zilnica). Tier 2 = cardificare mobil/tableta (speculativ).
|
||||
- **F3 (medium):** "CSS inline intr-un singur <style>, no build" e enuntat ca axioma, fara a fi
|
||||
cantarit vs alternative (Tailwind CDN, Pico.css). Pentru HTMX e o alegere aparabila, dar
|
||||
netestata/nelintata -> un blob `<style>` in crestere unde o editare de media-query strica tacit alte ecrane.
|
||||
- **F4 (critical, REZOLVAT partial):** risc de churn — re-implementare din proza. CEO recomanda
|
||||
recuperarea commit-ului revertit. **Verificat in git reflog: nu exista commit 5.13** (a fost
|
||||
explorare in working tree, revertita, niciodata commis). Deci nu e nimic de recuperat din git ->
|
||||
singura cale e re-implementarea din PRD -> PRD-ul TREBUIE facut mai specific (vezi Eng §2/§5).
|
||||
- **F5 (medium):** scenariul de regret la 6 luni — rewrite CSS multi-template pentru audienta cu
|
||||
mobil ~zero; `<style>` se dubleaza; un viitor content-change reintroduce clasa de bug "break vertical"
|
||||
pentru ca nimic structural nu o previne; modul "card pe tableta" nevazut de un user real. Ce NU va
|
||||
parea prost: design.md + sistemul de butoane.
|
||||
- **F6 (pozitiv):** design.md e bun (tokeni, ratii de contrast AA, gotcha CSS-var-in-@media). De
|
||||
pastrat/commis independent de soarta lui 5.13.
|
||||
|
||||
### CEO consensus table
|
||||
```
|
||||
CEO DUAL VOICES — CONSENSUS TABLE (codex N/A: usage limit)
|
||||
═══════════════════════════════════════════════════════════════
|
||||
Dimension Claude Codex Consensus
|
||||
──────────────────────────────────── ─────── ───── ─────────
|
||||
1. Premises valid? NO (F1) N/A 1-voce: NU (mobil nedovedit)
|
||||
2. Right problem to solve? PARTIAL N/A 1-voce: reframe la bug+butoane (F2)
|
||||
3. Scope calibration correct? NO N/A 1-voce: full P0+P1+P2 supradimensionat
|
||||
4. Alternatives explored? NO (F3) N/A 1-voce: framework nediscutat
|
||||
5. Competitive/market risks? N/A N/A irelevant (tool intern B2B)
|
||||
6. 6-month trajectory sound? PARTIAL N/A 1-voce: design.md+butoane da; cardificare risc
|
||||
═══════════════════════════════════════════════════════════════
|
||||
O singura voce -> nimic "CONFIRMED de ambele". Findings critice tratate ca atare oricum.
|
||||
```
|
||||
|
||||
### NOT in scope (confirmat / deferat)
|
||||
- Backend (worker, masina stari, idempotenta, contract RAR, canal API, mapping, validation): NEATINS. [PRD]
|
||||
- Tabele dense read-only (Jurnal/Nomenclator/Admin): raman scroll-contained, NU se cardifica. [PRD]
|
||||
- Refactor `routes.py`/`labels.py` pentru "Eroare/Eroare": deferat (debt). Dar vezi Design §2 — un guard
|
||||
de template (pill-only cand eticheta == stare) e ieftin si in-scope.
|
||||
- Adoptarea unui framework CSS: respins (P4 DRY + no-build potrivit stack-ului HTMX), dar de notat explicit ca decizie.
|
||||
|
||||
### What already exists (grounding pe cod real, nu pe "Raport VERIFY")
|
||||
- `docs/design.md` (232 linii): **FACUT**, calitate buna. Deliverabilul "design.md" e in esenta livrat.
|
||||
- Sistem butoane (`.act`/`.act-save`/`.btn-secondary`/`.btn-ghost`/`.btn-danger`/`.btn-sm`): **NEFACUT** (absent in base.html).
|
||||
- Macro `icon()` / `act_btn()` + Lucide: **NEFACUT** (absent in `_macros.html`; doar `camp`+`autosend_toggle`).
|
||||
- Stepper compact: **NEFACUT** (`_stepper.html` inca flex 4-coloane inline = exact anti-patternul P0).
|
||||
- Grid 2 carduri/rand tableta 768-1024px: **NEFACUT** (blocul @media 1024 doar ascunde `.col-actualizat`).
|
||||
- Card mobil Trimiteri (fara gap fix 120px): **NEFACUT** (`base.html:410-412` inca `td{display:flex}`+`::before{min-width:120px}`).
|
||||
- "Raport VERIFY ... PASS" din PRD = din explorarea revertita, NU stare curenta. De NU tratat ca acceptanta.
|
||||
|
||||
### Dream-state delta
|
||||
- CURRENT: 5.12 livrat (modal cont obligatoriu) + fundatie responsive 5.9/5.11; bug-uri P0 inca prezente in tree.
|
||||
- THIS PLAN: compactare + butoane + cardificare mobil/tableta + design.md (deja partial).
|
||||
- 12-MONTH IDEAL: un sistem de design tokenizat (design.md) aplicat consecvent, cu teste pe COMPORTAMENT
|
||||
la breakpoint (nu doar string-match pe clase), astfel incat editarile CSS sa nu mai poata reverti tacit.
|
||||
Delta: planul nu instituie teste pe comportament; ramane string-match fragil (vezi Eng §2).
|
||||
|
||||
### CEO completion summary
|
||||
Plan corect ca directie de design, dar (a) supra-incadrat ca "responsive" cand miezul de valoare e
|
||||
fix-bug-desktop + butoane + design.md; (b) premisa "mobil real" nedovedita; (c) re-implementare din
|
||||
proza fara specificul CSS/test -> risc mare de a reverti din nou. design.md e cel mai durabil activ.
|
||||
|
||||
---
|
||||
|
||||
## DECIZIE PREMISA (gate user, 2026-06-27)
|
||||
**Scope: pachet complet FARA grid 2-carduri/rand pe tableta.** Actionable lists (Trimiteri,
|
||||
Preview, Mapari) raman **o coloana pana la 1024px**. Reverseaza Decizie #2 din PRD.
|
||||
-> `design.md` §3 si §5.3 trebuie editate la "o coloana pana la 1024px" in acelasi PR.
|
||||
-> Deliverabilul "grid 2/rand" si testul lui aferent SCOT din scope; se adauga guard ca raman 1 coloana.
|
||||
|
||||
## Faza 2 — Design (subagent-only; codex usage-limited)
|
||||
|
||||
### Design dual voices
|
||||
- **Codex (design):** `[codex-unavailable: usage limit]`.
|
||||
- **Claude subagent (design):** independent. Findings:
|
||||
- **Meta-hazard (high):** sectiunea "Raport VERIFY ... PASS" se citeste ca raport de finalizare
|
||||
desi munca e nelivrata -> relabel "Criterii de acceptanta (de dovedit)" sau sterge.
|
||||
- **Ierarhie card mobil 7/10:** `vehicul=titlu, stare=pill, operatie+cod` corect. RISC pe ce se
|
||||
ascunde: (a) `actualizat` (timestamp) = singurul semnal "se misca / e blocat?" pe mobil ->
|
||||
pastreaza o linie meta 12px muted SAU garanteaza in modal si spune-o; (b) `checkbox` ascuns ->
|
||||
omoara multi-select pe mobil; daca bulk "Trimite la RAR" e workflow real = regresie functionala
|
||||
tacuta (high). PRD trebuie sa declare explicit daca bulk e desktop-only by design.
|
||||
- **Stari lipsa (high):** empty/loading/error/partial ale listelor cardificate nespecificate
|
||||
(fragmente HTMX-swapped). "Eroare/Eroare" (pill+eticheta) e cel mai vizibil exact in cardul nou
|
||||
-> guard de TEMPLATE ieftin (pill-only cand `eticheta_problema == stare_text`), nu refactor routes.py.
|
||||
- **Sistem iconite 8/10 (sound):** desktop-text/mobil-icon corect, rezolva "SVG invizibil in bloc".
|
||||
Gap: (a) `act_btn` TREBUIE sa emita `aria-label={{label}}` in ramura icon-only (invariant de macro);
|
||||
(b) Sterge fara confirmare pe 44px touch = risc data-loss la mis-tap -> confirm/undo pe `act-del`.
|
||||
- **Stepper 7/10:** "Pasul N din 4" + bara progres clar. CONTRADICTIE tableta: design.md §3 = tableta
|
||||
distincta, §5.4 baga "Tableta/mobil" impreuna in forma colapsata, iar PRD linia 5 zice "compact peste
|
||||
tot". La 820px = track slim sau "Pasul N din 4"? De ales explicit.
|
||||
- **Specificitate 9/10:** cel mai puternic punct; PRD referentiaza tokenii design.md. Contradictie minora:
|
||||
stepper inline foloseste `rgba(91,141,239,.10)` hardcodat vs design.md §2.1 (color-mix obligatoriu).
|
||||
|
||||
### Design litmus scorecard
|
||||
```
|
||||
DESIGN LITMUS (codex N/A: usage limit)
|
||||
═══════════════════════════════════════════════════════════════
|
||||
Dimensiune Claude Codex Nota
|
||||
────────────────────────────────── ────── ───── ──────────────
|
||||
1. Ierarhie informatie 7/10 N/A risc pe campuri ascunse (timestamp/checkbox)
|
||||
2. Stari (empty/loading/error) 3/10 N/A nespecificate (high)
|
||||
3. Sistem iconite/butoane 8/10 N/A aria-label macro + confirm delete
|
||||
4. Stepper / wizard 7/10 N/A contradictie tableta de rezolvat
|
||||
5. Specificitate + aliniere md 9/10 N/A 1 culoare hardcodata de scos
|
||||
6. Tabel->card responsive (2-up scos) N/A o coloana pana la 1024px (decizie user)
|
||||
7. Accesibilitate (design.md §6) 8/10 N/A solid; lipseste pattern confirm distructiv
|
||||
═══════════════════════════════════════════════════════════════
|
||||
```
|
||||
|
||||
### Auto-decizii Design (6 principii)
|
||||
- Guard template "Eroare/Eroare" (pill-only): **INCLUS** (P2 boil-lakes, <1 fisier, in cardul rescris). [TASTE -> gate]
|
||||
- `act_btn` invariant aria-label in ramura icon-only: **INCLUS** (P1 completeness + a11y obligatoriu). [mecanic]
|
||||
- Confirm/undo pe `act-del` mobil: **INCLUS** ca cerinta (P1; data-loss). Implementare = confirm nativ simplu, nu modal nou (P5). [TASTE -> gate]
|
||||
- Linie meta `actualizat` pe card mobil (12px muted, timp relativ): **INCLUS** (P1; semnal "blocat"). [TASTE -> gate]
|
||||
- Bulk-select pe mobil: **DECLARAT desktop-only by design** (P3 pragmatic; checkbox ascuns pe mobil ramane), de notat in PRD. [TASTE -> gate]
|
||||
- Relabel "Raport VERIFY" -> "Criterii de acceptanta": **INCLUS** (mecanic, evita falsa finalizare).
|
||||
- Rezolvare contradictie stepper tableta: **la 768-1024px = forma colapsata "Pasul N din 4"** (P5 explicit, consecvent cu mobil; track slim doar >=1024px). [mecanic]
|
||||
- Scoatere culoare hardcodata stepper -> `color-mix(var(--accent))`: **INCLUS** (DRY/tokeni design.md §2.1). [mecanic]
|
||||
|
||||
## Faza 3 — Eng (subagent-only; codex usage-limited)
|
||||
|
||||
### Eng dual voices
|
||||
- **Codex (eng):** `[codex-unavailable: usage limit]`.
|
||||
- **Claude subagent (eng):** independent, grounded pe cod real. Findings:
|
||||
- **§2 Fragilitate teste (CRITICAL):** vezi artefact test-plan. Invariantul PRD ("un singur bloc
|
||||
@media principal") e FACTUAL GRESIT — exista DOUA blocuri (377, 404); testele feliaza ferestre
|
||||
fixe `[idx:idx+5000]` de la PRIMA aparitie; rescrierea cardului impinge `min-height:0`/`100vw`
|
||||
peste fereastra -> `test_header...` + `test_modal...` PIC. Cauza probabila a revert-ului.
|
||||
FIX OBLIGATORIU INAINTE de CSS: refactor cele 2 teste sa ancoreze pe sentinel + slice pana la EOF.
|
||||
- **§1 Arhitectura (medium):** `act_btn` randeaza si textul si SVG-ul inline, ascunzand unul per
|
||||
breakpoint -> fiecare rand de tabel duce `<path>` Lucide chiar pe desktop (bloat DOM/octeti pe
|
||||
toate viewporturile). Acceptabil (P5 simplu) sau `<use href="#sprite">` definit o data.
|
||||
- **§3 Edge cases (medium):** VIN 17 car. (`_preview_rand.html:33` nowrap) — verifica sa nu produca
|
||||
scroll orizontal la 360px dupa scoaterea gap-ului 120px.
|
||||
- **§4 Teste (high):** `test_web_mapari_actiuni.py` are 3 aserturi pe `.icon-btn` (toate se rup);
|
||||
lipsesc teste pentru: card mobil fara 120px (P0 fara guard!), `act_btn` aria-label, stepper compact.
|
||||
Plus (post-decizie): guard ca actionable lists raman 1 coloana pana la 1024px (nu 2-up).
|
||||
- **§5 Risc re-implementare (high):** PRD da intentie, nu CSS exact/offset-uri; fara fix-ul de teste
|
||||
re-implementarea reverteaza din nou. `git reflog`: NU exista commit 5.13 de recuperat (explorare necommisa).
|
||||
- **§6 Complexitate ascunsa:** `_stepper.html` e ~70 linii inline -> mutarea in base.html = CRESTERE
|
||||
base.html; plaseaz-o DEPARTE de blocul mobil 404 ca sa nu strangi bugetul de octeti al ferestrei.
|
||||
|
||||
### Eng consensus table
|
||||
```
|
||||
ENG DUAL VOICES — CONSENSUS (codex N/A: usage limit)
|
||||
═══════════════════════════════════════════════════════════════
|
||||
Dimensiune Claude Codex Nota
|
||||
────────────────────────────────── ────── ───── ─────────────────
|
||||
1. Arhitectura sound? DA* N/A inline-CSS ok; SVG bloat minor
|
||||
2. Test coverage suficient? NU N/A fragil + lipsuri (high)
|
||||
3. Riscuri performanta? minor N/A SVG inline pe randuri
|
||||
4. Securitate? N/A N/A pur CSS/markup, fara suprafata noua
|
||||
5. Error paths? NU N/A stari card nespecificate (cu Design)
|
||||
6. Risc deploy/regresie? RIDICAT N/A revert auto daca testele nu se repara intai
|
||||
═══════════════════════════════════════════════════════════════
|
||||
```
|
||||
|
||||
### Arhitectura (diagrama)
|
||||
```
|
||||
base.html (un singur <style>, no build)
|
||||
├─ :root tokeni (design.md §2) [exista]
|
||||
├─ .btn / .btn-secondary/.btn-ghost/.btn-danger/.btn-sm [DE ADAUGAT]
|
||||
├─ .act / .act-save / .act-del / .act-group [DE ADAUGAT]
|
||||
├─ .stepper* (track slim >=1024; colapsat <1024) [DE ADAUGAT, departe de blocul 404]
|
||||
├─ @media (min-width:768px)and(max-width:1024px) { ... } [exista; FARA grid 2-up]
|
||||
├─ @media (max-width:767px) { #377 mic } [exista — sursa fragilitatii]
|
||||
└─ @media (max-width:767px) { #404 PRINCIPAL } [card rescris AICI, dupa header/modal]
|
||||
_macros.html : icon(name) + act_btn(label,ic,kind,attrs) [DE ADAUGAT; aria-label invariant]
|
||||
_stepper.html: rescris pe .stepper* (fara inline inalt) [DE RESCRIS]
|
||||
_mapari.html : .icon-btn -> act_btn [DE MIGRAT]
|
||||
TESTE : refactor ferestre fixe INAINTE de CSS [BLOCANT]
|
||||
```
|
||||
|
||||
### Auto-decizii Eng (6 principii)
|
||||
- Refactor cele 2 teste fragile INAINTE de CSS: **OBLIGATORIU, BLOCANT** (P1+P5; altfel revert garantat). [mecanic]
|
||||
- Corectare invariant PRD "un singur bloc @media": **CORECTAT** (sunt doua; regula reala = adauga dupa header/modal + slice EOF). [mecanic]
|
||||
- SVG inline vs `<use>` sprite: **inline acum** (P5 explicit/simplu); sprite notat ca optimizare. [TASTE -> gate]
|
||||
- Teste noi (#2 card fara 120px, #3 act_btn aria, #4 stepper, #5 window-guard, +1-coloana guard): **INCLUSE** (P1). [mecanic]
|
||||
- "Aserturile existente pastrate (toate trec)" din PRD: **CORECTAT** la "refactor + pastreaza intentia" (imposibil ca atare). [mecanic]
|
||||
|
||||
## Decision Audit Trail
|
||||
|
||||
| # | Faza | Decizie | Clasif. | Principiu | Rationament |
|
||||
|---|------|---------|---------|-----------|-------------|
|
||||
| 1 | CEO | Scope: complet FARA grid 2/rand tableta | GATE user | - | user a ales la premise gate; ambele voci design contestau 2-up |
|
||||
| 2 | CEO | Backend NEATINS, pur CSS/markup | mecanic | P4 | confirmat de PRD; zero schema |
|
||||
| 3 | CEO | Fara framework CSS (inline ramane) | mecanic | P4/P5 | no-build potrivit HTMX; notat ca decizie |
|
||||
| 4 | CEO | Commit revertit de recuperat? NU exista | mecanic | - | git reflog: explorare necommisa; re-impl din PRD |
|
||||
| 5 | Design | Guard template "Eroare/Eroare" pill-only | taste | P2 | in cardul rescris, <1 fisier |
|
||||
| 6 | Design | act_btn aria-label invariant icon-only | mecanic | P1 | a11y obligatoriu design.md §6 |
|
||||
| 7 | Design | Confirm/undo pe act-del mobil | taste | P1 | data-loss la mis-tap 44px |
|
||||
| 8 | Design | Linie meta `actualizat` pe card mobil | taste | P1 | singurul semnal "blocat" pe mobil |
|
||||
| 9 | Design | Bulk-select = desktop-only by design | taste | P3 | checkbox ascuns pe mobil; de declarat |
|
||||
| 10 | Design | Relabel "Raport VERIFY" -> "acceptanta" | mecanic | - | evita falsa finalizare |
|
||||
| 11 | Design | Stepper 768-1024 = forma colapsata | mecanic | P5 | consecvent cu mobil; track slim >=1024 |
|
||||
| 12 | Design | Stepper color-mix in loc de hardcodat | mecanic | P4 | design.md §2.1 tokeni |
|
||||
| 13 | Eng | Refactor teste fragile INAINTE de CSS | mecanic | P1/P5 | BLOCANT; cauza probabila revert |
|
||||
| 14 | Eng | SVG inline acum (sprite deferat) | taste | P5 | simplu > optimizare prematura |
|
||||
| 15 | Eng | 5 teste noi + migrare mapari | mecanic | P1 | acopera P0 + a11y + scope guard |
|
||||
| 16 | Eng | design.md §3/§5.3 -> "1 coloana <=1024" | mecanic | - | consecventa cu decizia de scope |
|
||||
|
||||
## Cross-Phase Themes
|
||||
- **"Raport VERIFY se citeste ca facut, dar nu e"** — semnalat de Design (meta-hazard) SI implicit de
|
||||
Eng (§5 re-impl). Semnal high-confidence: relabel + grounding pe cod real, nu pe raport.
|
||||
- **Stari/erori nespecificate** — Design §2 (stari card) + Eng §4 (error paths). De adaugat o matrice de stari.
|
||||
- **Fragilitate -> revert** — Eng §2 (teste) + CEO F5 (nimic structural nu previne re-bug). Repara testele intai.
|
||||
|
||||
## Eng completion summary
|
||||
Directia de design e fina; planul livreaza DOAR daca fragilitatea testelor (§2) e rezolvata INTAI
|
||||
(refactor cele 2 teste pe ancora+EOF), altfel rescrierea cardului mobil reverteaza singura din nou.
|
||||
Invariantul PRD despre "un singur bloc @media" e gresit si trebuie corectat. design.md ramane activul durabil.
|
||||
388
docs/prd/prd-5.14-mapare-llm-distilata.md
Normal file
388
docs/prd/prd-5.14-mapare-llm-distilata.md
Normal file
@@ -0,0 +1,388 @@
|
||||
<!-- plan sub /autoplan -->
|
||||
# PRD 5.14 — Mapare automata operatii service prin distilare LLM
|
||||
|
||||
**Stare**: inchis (2026-06-28; CLOSE dupa `/code-review high` -> embeddings „mort dar scump" reparat + WIRE functional la decizia user: corpus din nomenclator gated pe `AUTOPASS_EMBEDDINGS_ENABLED`; marime model corectata ~50MB->~230MB; regresie 1256 passed)
|
||||
|
||||
## Stories de executie (decompozitie lead, 2026-06-28)
|
||||
|
||||
> PRD-ul a fost aprobat prin /autoplan ca DESIGN (Decision Audit Trail #11-20). Aici lead-ul
|
||||
> il sparge in stories atomice executabile (ROADMAP §5.4), FARA a re-deschide deciziile.
|
||||
> **Secventiere fata de 5.15 (D9 + cerinta user "prioritate design 5.15"):** partile DISJUNCTE
|
||||
> de fisier ruleaza in PARALEL cu 5.15 acum; integrarea in editor (`mapping.py`/`routes.py`)
|
||||
> ASTEAPTA 5.15 si se aplica PESTE designul 5.15, fara sa-l suprascrie.
|
||||
|
||||
| Story | Tip | Fisiere (disjunct?) | Depinde de |
|
||||
|-------|-----|---------------------|-----------|
|
||||
| **L14-S1** Layer 1 etichetator offline | tool | `tools/mapare-llm/or_label.py` + teste (mock OpenRouter) — DISJUNCT | — |
|
||||
| **L14-S2** Temporal holdout (GATE Premisa 1) | tool | `tools/mapare-llm/holdout.py` + raport — DISJUNCT | — |
|
||||
| **L14-S3** Schema suggestions + shared store | backend | `app/schema.sql` (aditiv), store module nou, seeder, teste — owns schema.sql | — |
|
||||
| **L14-S4** Modul embeddings in-proces | backend | `app/embeddings.py` NOU + teste — DISJUNCT (modul; fara wiring) | — |
|
||||
| **L14-S5** Set held-out eval (BLOCANT auto-send) | tool | `tools/mapare-llm/heldout_eval.py` + metodologie — DISJUNCT | — |
|
||||
| **L14-S6** Integrare Layer 2/3 in editor | backend+UI | `app/mapping.py`, `app/web/routes.py` (editor) — **DUPA 5.15** | L14-S3,S4; 5.15 US-007/US-009 |
|
||||
|
||||
**Invariante de respectat (din Decision Audit Trail):** auto-send DOAR GOLD propriu (F-A/#11);
|
||||
silver in tabela SEPARATA, niciodata in resolve_prestatii (#13); seeder INSERT OR IGNORE, nu
|
||||
clobber uman (#2); scrub PII inainte de LLM (#3); NUL = ancore negative + supresie (#4);
|
||||
provenance source/confidence (#5); embeddings doar SUGESTIE + degradare gratioasa (#16b);
|
||||
held-out etichetat de OM = blocant pt orice auto-send peste GOLD (#19); tier "Inalta" sters din v1 (#17).
|
||||
|
||||
**Rezultat GATE Premisa 1 (L14-S2, 2026-06-28) — VERDICT: SLABA.** Validarea temporala STRICTA e
|
||||
imposibila (CSV-urile `docs/operatii-service/*.csv` au doar frecvente agregate, fara timestamp). Proxy
|
||||
Zipf + leave-first-out pe 155.195 operatii: pentru 90% acoperire de volum e nevoie de **4.368 denumiri
|
||||
distincte (25.4% din total)**, nu "cateva sute"; leave-first-out (limita superioara de stationaritate)
|
||||
= **88.9% agregat, SUB 90%**. Implicatie: etichetarea offline (L14-S1) trebuie sa proceseze ordine de
|
||||
MII de denumiri per client; coada `needs_mapping` ramane semnificativa chiar dupa bootstrap. Premisa nu e
|
||||
falsa, dar randamentul auto-rezolvarii e mai mic decat estima PRD-ul. NU blocheaza build-ul (piesele sunt
|
||||
utile + auto-send ramane conservator pe GOLD), dar recalibreaza asteptarile de acoperire. Tool: `tools/mapare-llm/holdout.py`.
|
||||
|
||||
**Raport VERIFY 5.14** (subagent independent context curat, 2026-06-28) — **VERDICT: PASS, zero FAIL,
|
||||
zero regresie 5.15.** `pytest -q -m "not live"` → **1245 passed, 0 failed**. Invariante confirmate cu cod+test:
|
||||
- **F1/#11/#17 auto-send DOAR GOLD propriu**: `load_mapping` citeste EXCLUSIV `operations_mapping` al
|
||||
contului; `resolve_prestatii` nu atinge DB (primeste `mapping` dict); singura cale spre `queued` =
|
||||
GOLD propriu. SILVER/GOLD-partajat/embedding = sugestie. Teste `test_f1_*` PASS. Tier "Inalta" sters (#17).
|
||||
- **#13 separare structurala**: grep confirma — `shared_store`/`mapping_suggestions`/`shared_mappings`
|
||||
apar DOAR in `enrich_suggestions` (apelat din `pending_unmapped`), niciodata in `resolve_prestatii`/`load_mapping`.
|
||||
- **#16b degradare gratioasa**: `is_available()=False` → `suggest_nearest=[]` fara exceptie; ingestia nu se blocheaza.
|
||||
- **#2** seeder INSERT OR IGNORE (nu clobber uman); **#4** NUL nu devine cod; **#5** provenance source/confidence;
|
||||
**#3** scrub PII nr/VIN inainte de LLM (`or_common.scrub`); **#19** held-out cu `cod_gold` GOL + kill-criterion
|
||||
(`wrong_code_rate<0.5%` AND `coverage>50%`) — toate PASS cu teste.
|
||||
- **GATE Premisa 1**: verdict **SLABA** documentat onest (proxy Zipf, fara pretentie de validare temporala).
|
||||
- fastembed 0.8.0 INSTALAT; testul real de embedding trece.
|
||||
|
||||
**Riscuri reziduale (LOW, non-blocant)**: (1) fastembed 0.8.0 foloseste mean-pooling (warning) — relevant doar
|
||||
daca se persista corpusul de vectori intre versiuni (acum re-indexat la nevoie din nomenclator); (2) `record_human_validation`
|
||||
ON CONFLICT nu suprascrie `cod_prestatie` (by design — corectie = override per-cont sau DELETE explicit);
|
||||
(3) lazy-load fastembed la prima cerere `/mapari` cand `AUTOPASS_EMBEDDINGS_ENABLED=true` (~230MB, cateva
|
||||
zeci de secunde daca modelul nu e in cache — acceptat la decizia CLOSE). **CLOSE 2026-06-28: embeddings WIRE-uit
|
||||
functional** (era „mort dar scump"): `ensure_embeddings_corpus(conn)` construieste corpusul din nomenclator
|
||||
(`nume_prestatie`->`cod_prestatie`), apelat in `pending_unmapped` + `_nemapate_pentru_submission` inainte de
|
||||
bucla, gated pe `AUTOPASS_EMBEDDINGS_ENABLED` (default OFF). Re-index doar la schimbarea semnaturii
|
||||
nomenclatorului. Corpusul se construieste din nomenclator (18 coduri largi), NU per-confirmare umana — sugestia
|
||||
embedding e similaritate denumire-prezentare vs. nume_prestatie RAR.
|
||||
|
||||
---
|
||||
|
||||
## Problema
|
||||
|
||||
La ingestie (canal API si import web), o prestatie poate veni cu `cod_op_service`
|
||||
+ `denumire` in loc de `cod_prestatie` RAR. Daca nu exista mapare, submission-ul
|
||||
intra in `needs_mapping` si asteapta confirmare umana. Service-urile reale au
|
||||
**volume mari de denumiri particulare** (masurat: 17.435 denumiri DISTINCTE in 4
|
||||
CSV-uri de clienti reali — `automotive` 13.170, `sigma` 3.743, `clever` 1.668,
|
||||
`south` 875). Maparea manuala a acestora, prin editorul `needs_mapping`, e
|
||||
prohibitiva: zeci de mii de operatii × confirmare umana.
|
||||
|
||||
Nomenclatorul RAR are doar **18 coduri** foarte largi (REPARATIE, INTRETINERE,
|
||||
REVIZIE PERIODICA, etc. — `nomenclator_seed.py`). Deci problema nu e potrivire de
|
||||
sinonime, ci **clasificare** a mii de operatii granulare in 18 categorii abstracte
|
||||
+ detectare de „gunoi" (linii care nu sunt operatii: `ITP CT 12 ABC`, `DISCOUNT
|
||||
MATERIALE 5%`, `MANOPERA`, nr. inmatriculare).
|
||||
|
||||
## Viziune (pivot 2026-06-28)
|
||||
|
||||
LLM-ul **NU ruleaza la runtime**. Rol unic: **etichetator offline** care
|
||||
construieste un set de date (denumire -> cod). La runtime ruleaza un **clasificator
|
||||
local mic, fara API** (similaritate / fuzzy / embeddings), „distilat" din etichetele
|
||||
LLM + maparile validate de oameni. Trei straturi:
|
||||
|
||||
1. **Etichetare offline (LLM, periodic):** acopera denumirile cu cele mai multe
|
||||
aparitii (frecventa) si grupeaza denumirile asemanatoare ca sa eticheteze ieftin.
|
||||
2. **Clasificator runtime (fara AI):** exact -> fuzzy/substring -> similaritate
|
||||
semantica (embeddings) peste baza de cunostinte. Zero cost per cerere, ruleaza pe
|
||||
LXC.
|
||||
3. **Baza de cunostinte PARTAJATA:** maparile validate de oameni din TOATE conturile
|
||||
de service contribuie la clasificare (strat „gold" comun), peste etichetele LLM
|
||||
(„silver" bootstrap). Munca de validare a unui service ajuta toate service-urile.
|
||||
|
||||
Viitor (nu acum): un LLM generativ local pe LXC. Pasul curent foloseste un model de
|
||||
**embedding** (nu generativ): mic, CPU, milisecunde/text.
|
||||
|
||||
## Premise
|
||||
|
||||
1. **Volumul de denumiri distincte e finit si se schimba lent.** Odata etichetate,
|
||||
90%+ din traficul viitor sunt repetari ale acelorasi denumiri (service-ul
|
||||
refoloseste propriul vocabular). Lege Zipf: top 100 denumiri = 43.6% volum,
|
||||
top 500 = 67.7%, top 1000 = 76.2% (din 155.195 operatii totale).
|
||||
2. **RAR accepta NUMAI coduri din nomenclator.** Un cod necunoscut -> HTTP 500
|
||||
(`ORA-12899`) + record PARTIAL FINALIZATA (terminal). Deci orice cod propus de
|
||||
un sistem automat TREBUIE validat fata de nomenclator inainte de enqueue
|
||||
(invariant existent in `resolve_prestatii(..., valid_codes)`).
|
||||
3. **Maparea gresita are cost asimetric:** un cod gresit trimis = FINALIZATA
|
||||
ireversibil la RAR. Deci pragul de auto-trimitere ramane conservator; incertul
|
||||
ramane `needs_mapping` cu om in bucla. Etichetele LLM NEVALIDATE = sugestie, nu
|
||||
auto-trimitere (vezi scara de incredere).
|
||||
4. **Hardware LLM local generativ e prea lent acum** (masurat: Ollama LXC 104
|
||||
generativ 180-320s/op). Embeddings locale insa sunt rapide pe CPU si suficiente
|
||||
pentru similaritate la runtime.
|
||||
5. **Datele nu sunt sensibile** (confirmat utilizator): denumirile de operatii pot
|
||||
merge la API-uri cloud pentru etichetare. PII incidental (nr. inmatriculare/VIN)
|
||||
se face scrub inainte de trimitere (F3).
|
||||
|
||||
## Masuratori
|
||||
|
||||
### Bootstrap (anterior, Groq)
|
||||
- Groq `llama-3.3-70b`: 28ms/op, acord 94% cu heuristica pe cazuri clare, detectare
|
||||
gunoi excelenta (`NUL`). Abandonat ca furnizor: cap zilnic free atins + cheie expusa.
|
||||
|
||||
### OpenRouter free — NVIDIA Nemotron (masurat 2026-06-28)
|
||||
Furnizor nou pentru etichetare: cheie utilizator, modele GRATUITE, date ne-sensibile.
|
||||
|
||||
- **Capcane de cont (rezolvate):** modelele free dau initial `404 No allowed
|
||||
providers` din cauza unui allowlist de provideri pe cont (venice/together/fireworks/
|
||||
atlas-cloud) — `open-inference`/`google-ai-studio`/`nvidia` erau excluse. Fix:
|
||||
eliminat restrictia in Settings -> Preferences + activat toggle-ul de privacy
|
||||
„free endpoints may publish/train". WAF: User-Agent `Mozilla/5.0` obligatoriu.
|
||||
- **Set fiabil = familia NVIDIA Nemotron.** Restul modelelor sunt 429 (rate-limited
|
||||
upstream, partajat global: llama-3.3-70b, qwen3-next, gemma, hermes, dolphin) sau
|
||||
404 (gpt-oss). Cap free tier ~50 cereri/zi fara credit.
|
||||
- **Test ensemble pe top 120 dupa frecventa (46.4% din volum), 2026-06-28:**
|
||||
|
||||
| Model | ms/op | parse-fail | acord vs Groq (overlap) |
|
||||
|---|---|---|---|
|
||||
| nemotron-3-super-120b | 1463 | 0 | 100% |
|
||||
| nemotron-nano-9b-v2 | 1248 | 0 | 100% |
|
||||
| nemotron-3-ultra-550b | 6450 | 0 | 100% |
|
||||
|
||||
Acord ensemble ponderat pe volum: **3/3 unanim = 87% volum**, 2/3 = 13%, dezacord
|
||||
total = **0%**. Din unanim: 7 NUL (gunoi), 100 coduri reale.
|
||||
- **Decizie model:** pastram `super-120b` + `nano-9b`; **aruncam `ultra-550b`**
|
||||
(4-5x mai lent, zero castig de acuratete). Caveat: ensemble din aceeasi familie
|
||||
NVIDIA -> acordul supraestimeaza increderea fata de un ensemble cross-family.
|
||||
- **Dezacordurile (13%) sunt cazuri de granita taxonomica reala**, nu zgomot:
|
||||
`REGLAT DIRECTIE/FARURI` (OE-2 intretinere vs OE-4 reglare), `MANOPERA
|
||||
TINICHIGERIE` (NUL vs OE-1), `DEZECHIPAT usa/bara` (pas de demontare), `INLOCUIT
|
||||
FILTRU AER` (OE-1 vs OE-3). Astea trebuie sa cada in `needs_mapping`.
|
||||
|
||||
## Solutia
|
||||
|
||||
### Stratul 1 — Etichetare offline (LLM, fara cod runtime)
|
||||
|
||||
Tool CLI (`tools/mapare-llm/`, stil `tools/apikey`). Etichetatorul OpenRouter
|
||||
(`or_common.py` + `or_label.py`) clasifica denumirile in cele 18 coduri RAR + `NUL`:
|
||||
|
||||
1. **Prioritizare pe FRECVENTA (NR), nu alfabetic.** Etichetam intai denumirile cu
|
||||
cele mai multe aparitii (acopera cel mai mult volum per apel).
|
||||
2. **Grupare pe similaritate inainte de etichetare.** Denumirile aproape identice
|
||||
(`REGLAT DIRECTIE` / `REGLAT DIRECTIA` / `REGLARE DIRECTIE`) se grupeaza; LLM
|
||||
eticheteaza doar **reprezentantul grupului**, codul se propaga la tot grupul.
|
||||
Maximizeaza acoperirea per apel LLM (critic pe cap free de ~50 cereri/zi).
|
||||
3. **Ensemble NVIDIA** (`super-120b` + `nano-9b`): acord -> incredere mai mare;
|
||||
dezacord -> ramane pentru `needs_mapping`. Vot pe coduri, nu self-confidence.
|
||||
4. **Scrub PII** (regex nr. inmatriculare/VIN) inainte de trimitere (F3, exista).
|
||||
5. Output: dataset etichetat cu `denumire, cod, sursa, confidence` (provenienta).
|
||||
`NUL` marcat separat (ancore negative + supresie), NU se promoveaza la cod RAR.
|
||||
|
||||
Prompt cu reguli explicite (avarii grave DOAR la accident; vopsire = reparatie;
|
||||
ulei+filtru = revizie; gunoi -> NUL). Batch mare (cap free tier), retry/backoff pe
|
||||
429, respecta `Retry-After`.
|
||||
|
||||
### Stratul 2 — Clasificator runtime (FARA AI, fara API)
|
||||
|
||||
Pentru o denumire din prezentare (canal API sau import), in `app/mapping.py`:
|
||||
|
||||
1. **Exact** in baza de cunostinte (`operations_mapping` + strat partajat) -> cod direct.
|
||||
2. **Fuzzy/substring** (`operation_text_rules`, `rapidfuzz`) — exista deja.
|
||||
3. **Similaritate semantica (embeddings)** — NOU: model multilingv mic (ex.
|
||||
`intfloat/multilingual-e5-small` sau `paraphrase-multilingual-MiniLM`), CPU.
|
||||
Vectorizam baza etichetata o data; la runtime vectorizam denumirea noua si luam
|
||||
cel mai apropiat vecin (sau top-k cu vot). Optional: clasificator `scikit-learn`
|
||||
(regresie logistica / kNN) antrenat pe (embedding -> cod) pentru generalizare
|
||||
dincolo de vecinul exact. „Antrenarea pe datele de test" = acest pas, secunde,
|
||||
ruleaza oriunde.
|
||||
4. Cod propus -> validat OBLIGATORIU `valid_codes` (garda ORA-12899). Peste pragul
|
||||
de incredere -> conform scarii; altfel `needs_mapping`.
|
||||
|
||||
Decizie de gazduire runtime: ramane deschisa pentru reviziile plan (in-proces in
|
||||
gateway vs microserviciu pe LXC/Flowise). Default propus: in-proces (cel mai simplu).
|
||||
|
||||
### Stratul 3 — Baza de cunostinte PARTAJATA cross-account
|
||||
|
||||
**Schimbare fata de versiunea anterioara** (care izola corpusul per cont):
|
||||
|
||||
- **Strat GOLD partajat:** maparile **validate de oameni** (din `needs_mapping`, in
|
||||
ORICE cont) intra intr-un store partajat `denumire_normalizata -> cod`. Astfel
|
||||
validarea facuta de un service ridica increderea pentru toate. Cheia = denumire
|
||||
normalizata (scrub PII, lower, strip), nu textul brut.
|
||||
- **Strat SILVER:** etichetele LLM (bootstrap) — sugestii, NU auto-trimitere.
|
||||
- **Override per-cont:** daca un cont mapeaza explicit o denumire la alt cod decat
|
||||
cel partajat (conflict legitim de vocabular), override-ul contului castiga pentru
|
||||
acel cont. Conflictele inter-cont se rezolva cu provenienta + (optional) majoritate.
|
||||
|
||||
Confirmarile umane curg organic prin folosirea normala a editorului `needs_mapping`
|
||||
— FARA sesiune separata de adjudecare manuala (cerinta utilizator).
|
||||
|
||||
### Scara de incredere (runtime, per operatie din prezentare)
|
||||
|
||||
| Treapta | Sursa | Actiune | Frictiune |
|
||||
|---|---|---|---|
|
||||
| Certa | exact in stratul GOLD (validat de om, orice cont) sau override cont | auto-trimite | zero |
|
||||
| Inalta | embedding NN cu similaritate FOARTE inalta la o mapare GOLD + ensemble LLM unanim | auto-trimite (prag calibrat) | zero |
|
||||
| Medie | LLM silver / similaritate medie | `needs_mapping` cu sugestie pre-completata -> 1 click | minima |
|
||||
| Joasa | similaritate slaba / coduri apropiate | `needs_mapping` manual | normala |
|
||||
| NUL | non-operatie (ITP, discount, nr. inmatriculare) | marcat „nu e operatie", suprimat | — |
|
||||
|
||||
**Invariant F1 (pastrat):** o eticheta pur-LLM NEVALIDATA nu auto-trimite singura;
|
||||
auto-send cere ori GOLD (validat de om), ori treapta „inalta" calibrata. Tensiunea
|
||||
centrala (utilizatorul se bazeaza pe LLM, dar FINALIZATA e ireversibil) = intrebarea
|
||||
cheie pentru reviziile plan: unde fix se aseaza bara treptei „inalta".
|
||||
|
||||
## Integrare
|
||||
|
||||
- Stratul 1: tool CLI offline `tools/mapare-llm/` (exista: `or_common.py`,
|
||||
`or_modeltest.py`; de adaugat `or_label.py` cu grupare + propagare).
|
||||
- Stratul 2: similaritate embeddings in `app/mapping.py` (`enrich_suggestions` ->
|
||||
`suggest_nearest`), apelata in `pending_unmapped` / `_nemapate_pentru_submission`
|
||||
pentru sugestia din editor. Corpusul se construieste din nomenclator via
|
||||
`ensure_embeddings_corpus` (gated pe `AUTOPASS_EMBEDDINGS_ENABLED`, default off):
|
||||
lazy-load model fastembed/ONNX (~230MB) la prima cerere /mapari cand flagul e activ,
|
||||
re-index doar la schimbarea nomenclatorului (semnatura). Off -> no-op (cade pe
|
||||
GOLD/SILVER + fuzzy). SUGGESTION-ONLY: NU intra in resolve_prestatii/enqueue (#13).
|
||||
- Stratul 3: store partajat (tabela noua `shared_mappings` sau coloana de scope pe
|
||||
`operations_mapping`), seed la confirmare umana; override per-cont.
|
||||
- Validare `valid_codes` pe tot lantul (exista).
|
||||
|
||||
## Non-obiective
|
||||
|
||||
- Nu inlocuim confirmarea umana pentru cazuri incerte.
|
||||
- Nu trimitem automat coduri sub prag / etichete LLM nevalidate.
|
||||
- Nu adaugam dependenta cloud la RUNTIME (LLM doar offline pentru etichetare).
|
||||
- Nu antrenam un LLM generativ local acum (viitor).
|
||||
|
||||
## Riscuri
|
||||
|
||||
- Etichete LLM gresite tratate ca adevar daca scapa garda F1 (seed direct in GOLD).
|
||||
- Ensemble aceeasi familie (NVIDIA) -> acord corelat-gresit; supraestimare incredere.
|
||||
- Strat partajat cross-account: o denumire poate insemna lucruri diferite la
|
||||
service-uri diferite -> conflict; mitigat prin override per-cont + provenienta.
|
||||
- Drift: denumiri noi neacoperite; embeddings ajuta dar nu elimina.
|
||||
- Free tier OpenRouter flaky (429/404, cap 50/zi) -> etichetarea bulk e lenta;
|
||||
e offline, deci tolerabil, dar nu pe calea critica de productie.
|
||||
- Model embedding ales: calitate pe limba romana de verificat empiric.
|
||||
|
||||
<!-- AUTONOMOUS DECISION LOG -->
|
||||
## Decision Audit Trail
|
||||
|
||||
| # | Faza | Decizie | Clasificare | Principiu | Rationament | Respins |
|
||||
|---|------|---------|-------------|-----------|-------------|---------|
|
||||
| 1 | Eng | Seed-ul NU intra direct in stratul auto-send; etichetele LLM = strat SILVER (sugestii). Auto-send cere GOLD (validat de om) sau treapta inalta calibrata | TASTE (critic) | P1, P5 | `resolve_prestatii`->`queued` direct => seed auto = AUTO-TRIMITERE ghiciri la FINALIZATA ireversibil (Premisa 3) | seed direct in auto-send |
|
||||
| 2 | Eng | Seeder = `INSERT OR IGNORE` / refuza overwrite pe randuri validate de om | MECHANICAL | P1 | re-rularea ar clobber-ui maparile umane cu ghiciri LLM | ON CONFLICT UPDATE |
|
||||
| 3 | Eng | Scrub regex (nr. inmatriculare/VIN) inainte de trimitere la LLM | TASTE | P1 | gunoiul contine `ITP CT 12 ABC` = nr. inmatriculare = PII | trimitere text brut |
|
||||
| 4 | Eng | NUL = ancore negative in corpus + lista supresie | MECHANICAL | P1 | altfel gunoiul recurent reintra mereu in needs_mapping si fuzzy ii da cod gresit | exclude NUL |
|
||||
| 5 | Eng | Coloana `source`/`confidence` (provenienta) pe baza de cunostinte | MECHANICAL | P1 | audit + rollback batch model prost + safe re-seed | fara provenienta |
|
||||
| 6 | Eng | Runtime = embeddings + clasificator mic (sklearn), NU LLM generativ | TASTE | P3, P5 | LLM generativ local prea lent (Premisa 4); embeddings CPU suficiente + rapide | LLM la runtime |
|
||||
| 8 | Eng | **SUPERSEDED:** corpus partajat cross-account (strat GOLD comun), NU per-cont izolat; override per-cont pe conflict | TASTE | P1, P2 | cerinta utilizator: validarea unui service ajuta toate; muncă compusa. Conflictul de vocabular rezolvat prin override + provenienta | (vechi: corpus strict per-cont) |
|
||||
| 9 | Eng | Furnizor etichetare = OpenRouter free, ensemble NVIDIA (super-120b + nano-9b); aruncat ultra-550b | MECHANICAL | P3 | masurat 2026-06-28: doar NVIDIA routeaza fiabil; ultra 4-5x lent fara castig | Groq (cap atins) / ultra |
|
||||
| 10 | Eng | Etichetare prioritizata pe frecventa + grupare pe similaritate (eticheteaza reprezentant, propaga) | MECHANICAL | P2 | acopera mult mai mult volum per apel; critic pe cap free ~50/zi | etichetare alfabetica |
|
||||
| 11 | CEO | **F-A: cross-account GOLD = suggestion-only**, nu auto-send cross-cont; doar GOLD PROPRIU (validat de omul contului) auto-trimite | GATE (user) | P1 | prima-intalnire cross-cont = FINALIZATA gresit ireversibil; override per-cont e post-hoc | cross-account auto-send (PRD scris) |
|
||||
| 12 | CEO | Premisa 1 (90% repeat) validata cu **temporal holdout INAINTE** de build | GATE (user) | P1 | concentrare-in-corpus != future-repeats-past; ieftin de verificat | build pe asumtie |
|
||||
| 13 | Eng | **Strat SILVER in TABELA SEPARATA** (mapping_suggestions), citita DOAR de suggest_codes/pending_unmapped; NICIODATA de load_mapping/resolve_prestatii | MECHANICAL | P5,P1 | scope-column pe operations_mapping auto-trimite silver (8+ call-site); separare structurala | scope column pe operations_mapping |
|
||||
| 14 | Eng | Shared store = tabela noua pe cheia `denumire_normalizata` (NU coloana pe operations_mapping: cheie diferita cod_op_service + UNIQUE) | MECHANICAL | P5 | spatii de chei diferite; conflict UNIQUE | scope column |
|
||||
| 15 | Eng | **Embeddings Layer 2 RAMANE in v1** (utilizatorul a respins amanarea la gate; mentine Decision #6). Recomandarea ambelor voci era amanare la v2 | USER CHALLENGE -> override user | P3,P5 | voci: 2GB pe ipoteza nemasurata, 18 clase acoperite de exact+fuzzy. User: vrea castig pe coada RO + control infra | (amanare v2) |
|
||||
| 16 | Eng | Embeddings = **IN-PROCES fastembed/ONNX** (~230MB pe disc, ONNX quantizat, fara torch; estimarea initiala de ~50MB a fost gresita — modelul multilingv `paraphrase-multilingual-MiniLM-L12-v2` are ~231MB chiar quantizat), in procesul API; model BAKED in imaginea Docker (sau volum cache) -> ZERO dependenta de retea la runtime. NU serviciu separat. Lazy-load la pornire, nu pe /healthz; worker NU incarca modelul | TASTE (user, revizuit) | P5,P3 | user: "embedding in interiorul aplicatiei, nu mai depind de alte resurse". Mai simplu + mai robust decat serviciu HTTP; ruleaza identic local si in Docker/LXC | serviciu separat Ollama/HTTP (revocat) / sentence-transformers+torch |
|
||||
| 16b | Eng | **Degradare gratioasa**: daca modelul nu se incarca -> ingestia NU se blocheaza, NU auto-trimite; cade pe exact+fuzzy, incertul -> needs_mapping. Embeddings raman doar SUGESTIE (consecinta F-A), in afara verdictului de enqueue (invariant dry-run/commit, Eng-F8) | MECHANICAL | P1 | esecul incarcarii modelului nu trebuie sa rupa coada; fara retea la runtime | block ingest pe model lipsa |
|
||||
| 17 | Eng | **Tier "Inalta" auto-send STERS din v1**; GOLD auto-trimite, restul (silver/NN/LLM-unanim) = needs_mapping 1-click | MECHANICAL | P1 | fara ground-truth; unanimitate same-family = eroare corelata, nu validitate | tier Inalta pe unanimitate LLM |
|
||||
| 18 | Eng | sklearn classifier scos din v1 | MECHANICAL | P5 | al doilea artefact antrenabil + pickle, castig marginal pe 18 clase | sklearn in v1 |
|
||||
| 19 | Eng | **Set held-out etichetat de OM = BLOCANT** pt orice tier auto-send peste GOLD propriu | MECHANICAL | P1 | "antrenare pe test" invalideaza orice precizie raportata | prag din etichete LLM |
|
||||
| 20 | CEO | OpenRouter: free OK pt bootstrap unic; credit mic ($5-20) pt drift steady-state (nu arhitecta pe cap 50/zi) | TASTE | P3 | juggling free > cost credit in timp eng | totul pe free tier |
|
||||
|
||||
## Istoric review (pre-pivot)
|
||||
|
||||
Versiunea anterioara a trecut prin `/autoplan` (mod SELECTIVE EXPANSION, subagent-only,
|
||||
Codex indisponibil). Constatari portante atunci: **F1 CRITIC** (seed=auto-send),
|
||||
F2/F3/F4 HIGH (idempotenta seed, scrub PII, ancore NUL), F5/F6/F7/F8 MEDIUM. Acele
|
||||
decizii sunt incorporate in Decision Audit Trail de mai sus. Pivotul 2026-06-28
|
||||
(LLM offline-only + runtime embeddings + strat partajat cross-account) NECESITA o
|
||||
noua rulare de review (CEO / Eng / Design) — de aceea sectiunea GSTACK REVIEW REPORT
|
||||
e goala momentan si se completeaza la urmatoarea rulare.
|
||||
|
||||
## GSTACK REVIEW REPORT
|
||||
|
||||
Rulat prin `/autoplan` 2026-06-28 (SELECTIVE EXPANSION). Voci: Claude subagent independent
|
||||
(CEO + Eng) + analiza orchestrator pe cod. **Codex INDISPONIBIL** (usage limit, reset 18 iul)
|
||||
-> mod single-reviewer. UI scope: NU (editorul needs_mapping exista deja). DX scope: borderline
|
||||
(CLI intern operator) -> Phase 3.5 sarit, considerente DX in Eng.
|
||||
|
||||
### Decizii GATE (confirmate de utilizator)
|
||||
- **F-A: cross-account = suggestion-only.** Maparile validate de orice cont PRE-COMPLETEAZA
|
||||
editorul needs_mapping (1-click) dar NU auto-trimit. Doar exact-match pe GOLD-ul PROPRIU
|
||||
(validat de omul contului) auto-trimite. Elimina riscul de FINALIZATA gresit cross-tenant.
|
||||
- **Premisa 1 validata cu temporal holdout INAINTE de build** (corpus primele N luni/client ->
|
||||
hit-rate exact pe lunile urmatoare). Ieftin, datele exista.
|
||||
|
||||
### Consens CEO (single-reviewer; Codex N/A)
|
||||
| Dimensiune | Claude | Verdict |
|
||||
|---|---|---|
|
||||
| Premise valide | NO (P1, P5) | flagged |
|
||||
| Problema corecta | PARTIAL | flagged |
|
||||
| Scope calibrat | NO (over-eng) | flagged |
|
||||
| Alternative explorate | NO | flagged |
|
||||
| Riscuri piata | PARTIAL | flagged |
|
||||
| Traiectorie 6 luni | AT RISK | flagged |
|
||||
|
||||
### Consens Eng (single-reviewer; Codex N/A)
|
||||
| Dimensiune | Claude | Verdict |
|
||||
|---|---|---|
|
||||
| Arhitectura | PARTIAL | flagged |
|
||||
| Acoperire teste | NO | flagged |
|
||||
| Footprint/perf | NO (2GB torch) | flagged |
|
||||
| Siguranta F1 | INTENT-OK | flagged |
|
||||
| Cai de eroare | PARTIAL | flagged |
|
||||
| Risc deploy | NO | flagged |
|
||||
|
||||
### Constatari portante (severitate)
|
||||
- **F-A / Eng-F1 (CRITIC):** auto-send DOAR pe GOLD. Strat SILVER in TABELA SEPARATA
|
||||
(`mapping_suggestions`), citita doar de suggest_codes/pending_unmapped, NICIODATA de
|
||||
load_mapping/resolve_prestatii. `auto_send` col e moarta (mapping.py:436); singura cale
|
||||
spre `queued` (auto-send, mapping.py:414) trebuie sa fie GOLD. Separarea = structurala.
|
||||
- **F-B (CRITIC):** toate masuratorile sunt ACORD (100% vs Groq, 87% unanim), nu ACURATETE
|
||||
vs ground-truth. Same-family NVIDIA = eroare corelata. Niciun tier auto-send peste GOLD
|
||||
pana nu exista set held-out etichetat de OM (esantion aleator stratificat).
|
||||
- **Eng-F2 (HIGH):** shared store pe cheia `denumire_normalizata` (NU `cod_op_service`) ->
|
||||
tabela noua obligatorie; precedenta override pinnata: account override > account GOLD >
|
||||
shared GOLD > text rules > unmapped.
|
||||
- **F-C / Eng-F3 (HIGH):** embeddings Layer 2 = over-engineering pe 18 clase Zipf-head.
|
||||
AMANAT v2. Daca se construieste: fastembed/ONNX (~230MB pe disc, ONNX quantizat;
|
||||
estimarea initiala de ~50MB a fost gresita), API-process-only, lazy, nu pe
|
||||
/healthz. NU in resolve_prestatii (altfel worker-ul ar avea nevoie de torch).
|
||||
- **Eng-F4 (HIGH):** tier "Inalta" sters din v1 (consecinta F-A + lipsa ground-truth).
|
||||
- **F-D (HIGH):** Premisa 1 nevalidata temporal -> gate (rezolvat).
|
||||
- **F-E (HIGH):** fara metrica de succes/baseline/kill-criterion -> de instrumentat
|
||||
(% linii auto-rezolvate la rata cod-gresit < 0.X%).
|
||||
- **MEDIUM:** NUL short-circuit inainte de suggest_codes + structura separata (Eng-F6);
|
||||
OpenRouter 429 resumabil + group radius conservator + provenance (Eng-F7); divergenta
|
||||
dry-run/commit (Eng-F8); credit mic vs free-tier (F-F); omisiune silentioasa NUL (F-G);
|
||||
calitate embedding RO de verificat (F-H); versionare cheie normalizare; drop sklearn v1.
|
||||
|
||||
### Teme cross-faza (semnalate independent in ambele faze)
|
||||
1. Auto-send DOAR GOLD; silver/embeddings/unanimitate-LLM = sugestie (CEO F-A/F-B + Eng F1/F4).
|
||||
2. Embeddings over-engineered pe 18 clase; amana sau fastembed (CEO F-C + Eng F3).
|
||||
3. Fara set ground-truth; masoara precizia inainte de orice tier auto-send (CEO F-B/F-E + Eng F4).
|
||||
|
||||
### NU in scope (amanat)
|
||||
- **sklearn classifier** peste embeddings (v2; embeddings raman doar NN suggestion in v1).
|
||||
- Orice tier auto-send peste exact-match GOLD propriu (pana la set held-out).
|
||||
- LLM generativ local la runtime (deja non-obiectiv PRD).
|
||||
- Tier "Inalta" calibrat (re-introdus doar cu eval cross-family + ground-truth).
|
||||
|
||||
**Embeddings Layer 2 RAMANE in v1** (override user la gate), IN-PROCES (fastembed/ONNX,
|
||||
model baked in imagine), DOAR sugestie, cu fallback gratios pe exact+fuzzy daca modelul nu
|
||||
incarca. Zero dependenta de retea la runtime. Vezi audit #15/#16/#16b.
|
||||
|
||||
### Ce exista deja (de refolosit, nu rescris)
|
||||
- `resolve_prestatii` / `classify_prezentare` / `reresolve_account` (mapping.py): precedenta
|
||||
cod direct > exact mapping > text rules > unmapped; garda valid_codes (ORA-12899).
|
||||
- `suggest_codes` (rapidfuzz token_sort) + `pending_unmapped`: punct de injectie sugestii.
|
||||
- `operation_text_rules` (substring) + `operations_mapping` (GOLD per-cont).
|
||||
- `tools/mapare-llm/` (or_common.py, or_modeltest.py) + pattern `*-partial.json` resumabil.
|
||||
- Scrub PII (F3), `normalize_for_match`, seed nomenclator (18 coduri).
|
||||
|
||||
### Artefact test plan
|
||||
`~/.gstack/projects/romfast-rar-autopass/mmarius-main-test-plan-20260628.md`
|
||||
(test F1-regression CRITIC + precedenta override + NUL + idempotenta seed + held-out eval).
|
||||
|
||||
### Stare review
|
||||
Aprobat prin `/autoplan` (vezi Decision Audit Trail #11-20 + #16b). Plan livrabil:
|
||||
v1 = Layer 1 (etichetare offline) + Layer 2 (embeddings ca SERVICIU SEPARAT configurabil,
|
||||
doar sugestie, fallback gratios) + Layer 3 (GOLD propriu auto-send + shared suggestion-only)
|
||||
+ exact/fuzzy existent + temporal holdout + metrica de succes + set held-out (blocant pt
|
||||
orice auto-send peste GOLD). v2 = sklearn classifier (dupa masurare).
|
||||
611
docs/prd/prd-5.15-propagare-design-dashboard-editare.md
Normal file
611
docs/prd/prd-5.15-propagare-design-dashboard-editare.md
Normal file
@@ -0,0 +1,611 @@
|
||||
# PRD 5.15 — Propagare design landing in aplicatie (dashboard compact + editare slim, VIN unic, prestatii multi-select)
|
||||
|
||||
**Stare**: inchis (2026-06-28; CLOSE dupa `/code-review high` -> 8 buguri reparate TDD; regresie 1256 passed, 1 deselected live; E2E browser real ramane OPEN — mediu sandbox fara Playwright)
|
||||
|
||||
> Proces complet: `docs/ROADMAP.md` §5. Contract RAR (sursa de adevar): `docs/api-rar-contract.md`.
|
||||
> Sistemul de design al landing-ului: `app/web/templates/landing.html` (commit 41aa385), `DESIGN.md`.
|
||||
> Mockup-uri piese fara design (REFERINTA VIZUALA OBLIGATORIE): `docs/mockups/prd-5.15-mockups.html`
|
||||
> — strip sanatate D6 (stari rosu/verde), picker prestatii E4 (op<->cod), reveal odometru initial.
|
||||
> Acopera exact piesele pe care mockup-urile landing nu le aratau si corecteaza contradictiile
|
||||
> mockup<->PRD (VIN unic, contor all-time, culori prin tokeni).
|
||||
> Starea trece: `draft -> aprobat -> in-executie -> verify-pass -> inchis`.
|
||||
|
||||
## 1. Obiectiv
|
||||
|
||||
Propagam sistemul de design al landing-ului comercial (carduri/liste/formulare compacte,
|
||||
slim, si cele 4 teme grafit/cobalt/cupru/hartie) in aplicatia reala. Concret: dashboard-ul
|
||||
Acasa primeste cardurile-contor + lista de trimiteri slim din mockup-ul hero, iar formularul
|
||||
de editare trimitere primeste designul compact din mockup-ul "prestatie noua", cu **un singur
|
||||
camp VIN**, **Observatii** ca text liber pentru operatiile de service si **prestatii ca chips
|
||||
multi-select** de coduri RAR. Userul a cerut explicit replicarea acestor doua mockup-uri pentru
|
||||
ca ii place cat de compacte/slim sunt.
|
||||
|
||||
Decizii de produs confirmate cu userul (poarta de aprobare a acestui PRD):
|
||||
- **D1**: cardurile-contor INLOCUIESC bara de status actuala (`_status.html`); pastram doar
|
||||
indicatorii de sanatate worker/RAR intr-o forma compacta.
|
||||
- **D2**: temele sunt ADITIVE — pastram light/dark/petrol + Auto SI adaugam cele 4 din landing
|
||||
(grafit/cobalt/cupru/hartie). Selectorul ciclic le parcurge pe toate. (grafit ~ dark si
|
||||
hartie ~ light raman optiuni separate, la cererea userului.)
|
||||
- **D3**: prestatiile sunt chips reale multi-select — utilizatorul poate adauga mai multe coduri
|
||||
din nomenclatorul RAR si poate sterge oricare; se trimite lista `prestatii` completa (RAR
|
||||
accepta lista `{codPrestatie, idPrezentare:null}` — `docs/api-rar-contract.md` §payload).
|
||||
- **D4** (contor Trimise): cardul "Trimise" arata trei valori temporale — **all-time** (principal)
|
||||
+ **luna asta** + **azi** (secundar). Necesita extinderea numaratorilor cu `sent_today`/`sent_month`.
|
||||
- **D5** (Observatii = operatii service): in API-ul RAR, campul `obs` e DE FAPT denumirea operatiilor
|
||||
din service. Deci `obs` = text liber cu operatiile efectuate; la import, daca fisierul nu are
|
||||
coloana Observatii, **concatenam denumirea operatiei de service in `obs`**. `obs` ramane in
|
||||
`payload_json` (camp din contractul RAR), fara coloana noua.
|
||||
|
||||
Decizii din /plan-ceo-review (2026-06-28, mod SELECTIVE EXPANSION):
|
||||
- **D6** (sanatate mereu-vizibila): cardurile-contor inlocuiesc bara de status, DAR sanatatea
|
||||
(worker viu? RAR accesibil? ultima autentificare) ramane intr-un **strip mereu-vizibil, colorat,
|
||||
deasupra contoarelor** (verde "declaratiile curg" / rosu "blocat: worker oprit / RAR inaccesibil").
|
||||
Invariant: zero-silent-failures — semnalul critic NU se ingroapa sub volum. (Rafineaza D1.)
|
||||
- **D7** (operatie -> obs, fara regresie de mapare): la import, denumirea operatiei RAMANE in
|
||||
`op_service` (sursa pentru maparea op->cod) SI se COPIAZA in `obs`. `obs` e sink aditional, nu
|
||||
mutare; fluxul needs_mapping ramane neatins. (Rafineaza D5.)
|
||||
- **D8** (idempotenta obs): `obs` e EXCLUS din cheia de idempotenta (`idempotency.py:98`). Deci
|
||||
editarea `obs` NU schimba cheia si NU poate crea duplicate — corecteaza AC-ul gresit din US-005.
|
||||
`prestatii` ESTE in cheie (sortat dupa cod) — multi-select re-cheieaza randul (US-006).
|
||||
- **D9** (secventiere): 5.15 INAINTE de 5.14 (mapare LLM). Editorul manual defineste forma listei
|
||||
`prestatii` si UX-ul de confirmare; 5.14 umple codurile peste aceeasi forma.
|
||||
- **D10** (extinderi acceptate, SELECTIVE EXPANSION): toate 4 intra in scope — (a) salvare mapare
|
||||
din chip (US-009), (b) bulk-fix din lista (US-010), (c) require dinamic odometruInitial la chip
|
||||
R-ODO/I-ODO (US-007), (d) editare keyboard-first in form slim (US-007).
|
||||
|
||||
Decizii din /plan-eng-review (2026-06-28, model claude/opus; outside-voice = Claude subagent,
|
||||
Codex a atins usage-limit). Fiecare confirmata cu userul:
|
||||
- **E1** (ARCH, /repune nu mai sterge operatia): `/repune` face azi `p0.pop("cod_op_service")`
|
||||
la `routes.py:1326` — sterge operatia cand se seteaza un cod direct, rupand D7 si US-009.
|
||||
US-006 ELIMINA acel `pop` si pastreaza `cod_op_service`; test de regresie obligatoriu
|
||||
(op_service supravietuieste unui /repune cu cod). (Rafineaza US-006.)
|
||||
- **E2** (DRY teme, fisier fierbinte): config-ul de teme e duplicat in ~7 locuri in base.html
|
||||
(anti-FOUC `VALID` la :22 + cinci literali paraleli `CYCLE`/`VALID`/`ICONS`/`LABELS`/`NEXT`
|
||||
la :758-765). US-001 CONSOLIDEAZA intr-o singura structura sursa-de-adevar (`THEMES`
|
||||
ordonata) din care se DERIVA ciclul/etichetele/iconitele + setul anti-FOUC. Adaugarea unei
|
||||
teme = o intrare. (Rafineaza US-001.)
|
||||
- **E3** (obs concat idempotent): la import, copierea denumirii operatiei in `obs` se face
|
||||
DERIVE-ON-EMPTY (doar cand `obs` e gol) ca sa fie idempotenta la re-import/re-editare. Test
|
||||
dedicat anti-dublu-concat ("Schimb ulei; Schimb ulei"). (Rafineaza US-005.)
|
||||
- **E4** (binding operatie<->cod in chips — HIGH): chip-urile NU sunt o lista plata de coduri.
|
||||
Cand exista operatii (`cod_op_service`), UI-ul randeaza UN picker PE operatie (eticheta op +
|
||||
chip-ul ei de cod), pastrand perechea per-item pe care modelul o are deja; lista plata de
|
||||
coduri libere DOAR pentru cazul fara operatie (corectie pura). Astfel US-009 citeste perechea
|
||||
direct, iar deduparea e PER-ITEM (nu "dupa cod" — doua operatii distincte pot mapa legitim la
|
||||
acelasi cod RAR). (Rafineaza US-006 AC2 + US-007 AC3 + US-009.)
|
||||
- **E5** (serializare Val 3 pe routes.py): US-005 si US-006 rescriu ACEEASI functie
|
||||
`post_corecteaza` (`routes.py:1120-1262`). Regula "un singur autor pe fisier fierbinte" se
|
||||
EXTINDE la routes.py in Val 3: US-005 INAINTE de US-006 (secvential, nu paralel). (Rafineaza §6.)
|
||||
- **E6** (US-007 HTMX server-driven PRIMARY): inversam abordarea — chips add/remove via `hx-post`
|
||||
care re-randeaza partial-ul chips+form; reveal-ul conditional `odometruInitial` rezulta GRATIS
|
||||
din re-randarea server; navigare tastatura = `<select>`/`<datalist>` nativ. JS custom DOAR ca
|
||||
progressive enhancement (snappiness), nu calea principala. Elimina path-ul dublu JS/no-JS.
|
||||
(Rafineaza US-007.)
|
||||
- **E7** (contoare in timp local RO): `azi`/`luna asta` se bucketeaza in timp local RO (UTC+2/+3),
|
||||
nu UTC — `updated_at` e `datetime('now')` UTC, deci `date(updated_at)` pur ar numara gresit
|
||||
trimiterile dintre miezul noptii local si ~03:00. Folosim offset RO (ex. `date(updated_at,'+3 hours')`
|
||||
cu aceeasi baza `now`) + test la granita de miez de noapte local. (Rafineaza US-003.)
|
||||
- **E8** (interleave fix authz GET-listari — securitate): CLAUDE.md noteaza scurgere cross-cont
|
||||
deschisa ("GET-urile de listare sunt globale + neprotejate"). Userul a ales sa INTERLEAVE
|
||||
remedierea in 5.15 -> story noua **US-011** (account-scope pe GET-urile de listare + teste),
|
||||
nu queue dupa polish-ul de teme.
|
||||
|
||||
Fapte verificate care fundamenteaza scope-ul (nu presupuneri):
|
||||
- `vin` la RAR e **un singur camp** (17 car., MAJUSCULE, fara O/I/Q) — cerinta "fara 2 campuri
|
||||
VIN" e deja respectata azi (`_form_editare.html` are un singur `vin`); ramane sa NU regresam.
|
||||
- `prestatii` e deja **lista** in modelul intern (`mapping.resolve_prestatii(prestatii: list[dict])`)
|
||||
si in contractul RAR — multi-select nu cere model nou, ci editor nou.
|
||||
- `obs` exista deja ca alias de coloana la import (`import_router.py:71` — Observatii/Obs/Mentiuni/Note)
|
||||
si ca text liber optional in contractul RAR (`obs`); azi NU e editabil in formular.
|
||||
|
||||
## 2. Non-Goals (anti scope-creep)
|
||||
|
||||
- Fara modificari pe backend-ul de trimitere: worker, masina de stari, idempotenta-logica
|
||||
(`build_key`), reconciliere, contract RAR. Recalcularea idempotentei la editare foloseste
|
||||
mecanismul EXISTENT (ca la 3.5/5.10), nu unul nou.
|
||||
- Fara migrare de schema decat daca strict necesar. `obs` si `prestatii` traiesc in
|
||||
`submissions.payload_json` (de confirmat la US-005) — fara coloane noi daca payload-ul le poarta.
|
||||
- Fara stergerea functionalitatii listei de trimiteri: filtre (data/vehicul/stare), paginare,
|
||||
bulk-delete pe randuri blocate, click->detaliu raman; se schimba DOAR aspectul randului (slim).
|
||||
- Fara schimbarea regulilor de mapare operatie->cod sau a validarii nomenclatorului RAR
|
||||
(`mapping.py`, `validation.py` raman ca atare; doar callsite-urile de editare le folosesc cu lista).
|
||||
- Fara redesign al landing-ului (deja livrat in 5.x); aici doar IMPORTAM stilul lui in app.
|
||||
|
||||
## 3. Stories atomice
|
||||
|
||||
> Backend + UI pentru acelasi comportament = 2 stories. `base.html` e fisier FIERBINTE
|
||||
> (serializat intre valuri — un singur autor pe val). Toate UI verificate pe un esantion de teme (o tema luminoasa + una intunecata).
|
||||
|
||||
### US-001: Teme aditive (light/dark/petrol + grafit/cobalt/cupru/hartie) + tokeni `--card2`/`--line2`
|
||||
**Ca** operator de service **vreau** aceleasi teme ca pe landing **pentru ca** aplicatia sa para
|
||||
acelasi produs, coerent vizual.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/templates/base.html`, `DESIGN.md`, `tests/test_tema.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_tema.py` — `test_cele_4_teme_definite`, `test_tokeni_card2_line2_in_toate_temele`, `test_anti_fouc_4_stari`, `test_migrare_localStorage_legacy`
|
||||
- **Acceptance criteria**:
|
||||
- [x] Pastram temele EXISTENTE light/dark/petrol si ADAUGAM 4 teme noi grafit/cobalt/cupru/hartie,
|
||||
definite prin token-urile EXISTENTE (`--bg/--card/--ink/--muted/--line/--ok/--warn/--err/--accent`)
|
||||
+ DOUA noi `--card2` (fundal input/contor) si `--line2` (separator subtire). `--card2`/`--line2`
|
||||
primesc valori si in light/dark/petrol (fallback rezonabil). Maparea landing->app pentru cele 4
|
||||
noi: `--text->--ink`, `--sub->--muted`, `--okt->--ok`, `--errt->--err`, `--infot->--accent`.
|
||||
- [x] Selectorul ciclic parcurge TOATE: light -> dark -> petrol -> grafit -> cobalt -> cupru ->
|
||||
hartie -> Auto, afiseaza eticheta temei curente, persistenta `localStorage` (D2).
|
||||
- [x] **DRY (E2)**: config-ul de teme traieste intr-o SINGURA structura sursa-de-adevar
|
||||
(`THEMES` ordonata, cu `{id,label,icon}`) din care se DERIVA `CYCLE`/`NEXT`/`ICONS`/`LABELS`
|
||||
(azi 5 literali paraleli la base.html:758-765) SI setul anti-FOUC `VALID` (azi separat la
|
||||
base.html:22). Adaugarea unei teme noi = o singura intrare; test ca derivatele acopera
|
||||
toate temele (prinde o intrare ICONS/LABELS lipsa, nu doar token CSS lipsa).
|
||||
- [x] "Auto" pastrat: urmeaza `prefers-color-scheme`, rezolva la dark/grafit sau light/hartie
|
||||
(decizie minora: Auto -> dark + hartie pentru light, sau dark/grafit — aliniaza cu I2).
|
||||
- [x] Script anti-FOUC in `<head>` seteaza `data-theme` sincron pre-paint pentru toate starile;
|
||||
valoare necunoscuta -> Auto, fara blink. Valorile vechi raman valide (nu se mapeaza fortat).
|
||||
- [x] Contrast AA pentru text principal in toate temele (light + hartie sunt cele luminoase).
|
||||
- [x] `DESIGN.md` actualizat: sectiunea cromatica + selector tema reflecta toate temele.
|
||||
- **Verificare E2E**: browser pe `/` (dashboard logat) — ciclare prin toate temele, persistenta la
|
||||
refresh, fara FOUC; toate temele selectabile.
|
||||
|
||||
### US-002: Componente de design slim in `base.html` (CSS, fara consumatori inca)
|
||||
**Ca** dezvoltator **vreau** clase reutilizabile pentru carduri-contor, lista slim, campuri slim si
|
||||
chips **pentru ca** dashboard-ul si formularul sa le consume DRY, identic cu mockup-ul.
|
||||
|
||||
- **Depinde de**: US-001 (foloseste `--card2`/`--line2`)
|
||||
- **Fisiere**: `app/web/templates/base.html`, `DESIGN.md`, `tests/test_web_responsive.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_responsive.py` — `test_clasa_contor_card`, `test_clasa_lista_slim`, `test_clasa_camp_slim`, `test_clasa_chips`
|
||||
- **Acceptance criteria**:
|
||||
- [x] `.contor-card` (sau nume aliniat conventiei): cifra mare bold + eticheta mica muted, fundal
|
||||
`--card2`, bordura `--line`, radius 8px, padding 10-12px; variante de culoare a cifrei prin
|
||||
`.s-*` existente (verde/accent/rosu).
|
||||
- [x] `.lista-trimiteri-slim` cu rand `.trimitere-slim`: stanga = VIN mono (linia 1) + operatie·ora
|
||||
muted (linia 2, 11px); dreapta = pill de stare; separator `--line2`; padding 10-14px.
|
||||
Randul ramane clickabil (rol button) si pastreaza tinta 44px pe mobil.
|
||||
- [x] Varianta slim de camp formular: label 11px muted deasupra, input ~30px inaltime, fundal
|
||||
`--card2`, mono pentru VIN/odometru/nr; integrata in macro-ul `camp` din `_macros.html`
|
||||
printr-un flag (`slim=True`), fara a rupe randarea actuala (default neschimbat).
|
||||
- [x] `.chips` + `.chip` (cu buton `×` de stergere) pentru prestatii multi-select; accesibil
|
||||
(buton real cu `aria-label`), stilat ca in mockup (accent 18%, font 10-11px).
|
||||
- [x] **Doar tokeni, fara hex hardcodat (criteriu din mockup)**: toate culorile componentelor noi
|
||||
(contor, lista slim, chips, strip, picker) folosesc EXCLUSIV variabile CSS
|
||||
(`var(--errt)`/`var(--okt)`/`var(--accent)`/`var(--card2)`/`var(--line2)` etc.), NU hex literal
|
||||
si NU inline-styles copiate ca-atare din `landing.html`. Cifra "De corectat" rosie = token
|
||||
(`var(--errt)`), nu `#E05D5D` hardcodat, ca sa ramana AA pe temele luminoase (hartie/light).
|
||||
Referinta: `docs/mockups/prd-5.15-mockups.html`.
|
||||
- [x] Zero regresie vizuala pe componentele existente (`.card/.pill/.act/.tabel-trimiteri`).
|
||||
- **Verificare E2E**: pagina de proba/sandbox sau direct in US-003/004/007; vizual pe un esantion de teme + 390/1280.
|
||||
|
||||
### US-003: Dashboard Acasa — carduri-contor inlocuiesc bara de status
|
||||
**Ca** operator **vreau** cele 3 carduri-contor compacte (Trimise / In coada / De corectat)
|
||||
**pentru ca** sa vad starea dintr-o privire, ca in mockup.
|
||||
|
||||
- **Depinde de**: US-002
|
||||
- **Fisiere**: `app/web/templates/_status.html`, `app/web/templates/_acasa.html`,
|
||||
`app/web/routes.py` (`_status_counts` extins cu `sent_today`/`sent_month`), `tests/test_web_status.py`,
|
||||
`tests/test_web_dashboard.py` (~5 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_status.py` — `test_strip_sanatate_mereu_vizibil`, `test_strip_rosu_worker_oprit`, `test_trei_contoare_card`, `test_trimise_all_time_luna_azi`, `test_fara_bara_veche`
|
||||
- **Acceptance criteria**:
|
||||
- [x] **Strip de sanatate mereu-vizibil, DEASUPRA contoarelor** (D6): o linie compacta colorata —
|
||||
verde "declaratiile curg" cand worker viu + RAR ok; **rosu** + text explicit cand worker
|
||||
oprit SAU RAR inaccesibil ("Blocat: worker oprit" / "Blocat: RAR inaccesibil"), cu ultima
|
||||
autentificare RAR. Glife accesibile ✓/✗ (nu doar culoare). Invariant zero-silent-failures:
|
||||
semnalul "declaratiile NU pleaca" e imposibil de ratat, NU ingropat sub volum.
|
||||
**Layout exact (din mockup)**: strip full-width sus; glifa (✗ rosu / ✓ verde) + text bold la
|
||||
stanga, "Ultima autentificare RAR: ..." mono muted la dreapta. Copy: rosu "Blocat: worker oprit
|
||||
— declaratiile NU pleaca" (sau "... RAR inaccesibil"), verde "Declaratiile curg normal".
|
||||
Referinta: `docs/mockups/prd-5.15-mockups.html`.
|
||||
- [x] Sub strip: card "Trimiteri RAR AUTOPASS" cu 3 contoare slim: **In coada** (queued, accent),
|
||||
**Trimise** (sent, verde), **De corectat** (blocate = needs_data + needs_mapping + error, rosu).
|
||||
- [x] **Stari goale + ierarhie contor (criteriu din mockup)**: cifra principala a contorului "Trimise"
|
||||
e **all-time** (cifra mare bold), iar "luna asta"/"azi" sunt o sub-linie mono secundara
|
||||
(`luna {n} · azi {n}`) — NU "luna asta" ca cifra principala (corecteaza framing-ul din mockup-ul
|
||||
landing). Contorul "De corectat" la 0 se afiseaza **muted, nu rosu** (rosu doar cand exista
|
||||
blocate — pastreaza pattern-ul `_status.html:47`). Referinta: `docs/mockups/prd-5.15-mockups.html`.
|
||||
- [x] Cardul **Trimise** afiseaza trei valori temporale (D4): all-time (cifra principala) + "luna asta"
|
||||
+ "azi" (sub-linie secundara). `_status_counts` extins cu `sent_today`/`sent_month`.
|
||||
**Sursa de timp**: NU exista coloana `sent_at`; folosim `status='sent' AND date(updated_at)=...`.
|
||||
Justificare (verificat): un rand `sent` nu mai primeste scrieri ulterioare pana la purge-delete
|
||||
la +90z (`purge_after` se seteaza in ACEEASI scriere care marcheaza `sent`), deci `updated_at`
|
||||
== momentul trimiterii pentru randurile `sent` -> fara migrare de coloana (respecta Non-Goal).
|
||||
Daca pe viitor apar scrieri post-`sent`, reevalueaza o coloana `sent_at` dedicata.
|
||||
**Timezone (E7)**: `updated_at` e `datetime('now')` = UTC; bucketarea `azi`/`luna asta`
|
||||
se face in TIMP LOCAL RO (ex. `date(updated_at,'+3 hours')`, aceeasi baza `now`), altfel
|
||||
trimiterile dintre miezul noptii local si ~03:00 cad pe ziua precedenta si "luna asta" e
|
||||
gresita in primele ore ale zilei de 1. Test la granita de miez de noapte local.
|
||||
**Caveat reconcile (E6 outside-voice)**: pe reconciliere (raspuns pierdut) worker-ul
|
||||
marcheaza `sent` cu `updated_at` = momentul reconcilierii, nu al inserarii RAR — pentru
|
||||
randurile reconciliate (rare) `updated_at` poate diferi de momentul real al trimiterii.
|
||||
- [x] Navigarea existenta (Trimiteri/Mapari + badge needs_mapping) se pastreaza. Click pe contorul
|
||||
**De corectat** deep-link-eaza in lista filtrata pe blocate (`?status=` existent din 5.x),
|
||||
nu intr-o pagina noua.
|
||||
- [x] Scoped pe cont; poll-ul existent (`/_fragments/status`) randeaza noul antet fara a pierde tab-ul.
|
||||
- [x] Responsive: cele 3 contoare pe un rand pe desktop, stivuite/2-pe-rand pe mobil, fara overflow.
|
||||
- **Verificare E2E**: browser pe `/` — contoare corecte vs date din DB, sanatate worker mort/viu,
|
||||
poll pastreaza starea.
|
||||
|
||||
### US-004: Lista de trimiteri — rand slim (VIN + operatie·ora + pill)
|
||||
**Ca** operator **vreau** lista de trimiteri in stil slim ca in mockup **pentru ca** e mai compacta
|
||||
si mai usor de scanat, pastrand filtrele si actiunile.
|
||||
|
||||
- **Depinde de**: US-002
|
||||
- **Fisiere**: `app/web/templates/_submissions.html`, `app/web/templates/_coada.html` (filtre raman),
|
||||
`tests/test_web_submissions.py`, `tests/test_web_responsive.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_submissions.py` — `test_rand_slim_vin_operatie_pill`, `test_filtre_paginare_pastrate`, `test_bulk_doar_blocate`, `test_click_deschide_detaliu`
|
||||
- **Acceptance criteria**:
|
||||
- [x] Fiecare rand: stanga VIN mono scurt (`vin_scurt`) linia 1 + operatie + ora/data muted linia 2;
|
||||
dreapta pill de stare (`stare_css`/`stare_scurt`). Nr. inmatriculare, data completa si nr.
|
||||
prezentare RAR raman accesibile (linie meta discreta si/sau in modalul de detaliu).
|
||||
- [x] Filtre (data/vehicul/stare — `_coada.html`), paginarea numerotata si bulk-delete pe randuri
|
||||
blocate (checkbox doar pe `gestionabil`) raman FUNCTIONALE.
|
||||
- [x] Click pe rand deschide `/_fragments/trimitere/{id}` in modal (neschimbat).
|
||||
- [x] Slim layout consistent desktop si <=1024px (cardurile responsive existente nu regreseaza).
|
||||
- [x] Pill-urile de stare folosesc maparea din `labels.py` (zero etichete noi). Eticheta "Eroare VIN"
|
||||
din mockup-ul landing e DOAR ilustrativa — se foloseste `stare_scurt` existent (ex. "De corectat").
|
||||
- **Verificare E2E**: browser — filtrare + paginare + click detaliu + bulk pe blocate, pe 4 teme,
|
||||
pe 390/820/1280.
|
||||
|
||||
### US-005: Backend — `obs` (Observatii) editabil si persistat
|
||||
**Ca** operator **vreau** sa editez Observatiile (operatiile de service in text liber) **pentru ca**
|
||||
sa corectez/completez ce s-a facut, separat de codurile RAR.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/routes.py` (`/trimitere/{id}/corecteaza`),
|
||||
`app/api/v1/import_router.py` (`/_import/{id}/rand/{row}/editeaza`, `EDIT_FIELDS`),
|
||||
`app/validation.py` (obs optional), `app/payload_view.py` (echo obs),
|
||||
`tests/test_web_corectie*.py`, `tests/test_import_review.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_corectie_obs.py` — `test_obs_editabil_persistat_corecteaza`, `test_obs_persistat_preview_editeaza`, `test_obs_optional_gol_ok`, `test_import_concateneaza_operatie_in_obs`
|
||||
- **Acceptance criteria**:
|
||||
- [x] `obs` traieste in `payload_json` (camp `obs` din contractul RAR); fara coloana noua / migrare (D5).
|
||||
- [x] `obs` adaugat in `EDIT_FIELDS`; `corecteaza` si `editeaza` (preview) accepta si persista `obs`.
|
||||
- [x] `obs` optional (text liber, fara validare de continut, doar trim); apare in `payload_view`.
|
||||
- [x] `obs` se include in payload-ul trimis la RAR (camp `obs`). **`obs` e EXCLUS din cheia de
|
||||
idempotenta** (`idempotency.py:98`) — deci editarea DOAR a `obs` NU schimba cheia si NU poate
|
||||
crea duplicat (D8). NU recalcula/forta cheia pe baza `obs`. (Corecteaza formularea anterioara.)
|
||||
- [x] **La import** (D7): denumirea operatiei RAMANE in `op_service` (sursa pentru maparea op->cod);
|
||||
daca fisierul NU are coloana Observatii, denumirea operatiei se **COPIAZA** (nu se muta) si in
|
||||
`obs`; daca are coloana Observatii, se pastreaza textul ei. Format de concatenare definit
|
||||
(denumiri separate prin "; "). Fluxul needs_mapping ramane neatins.
|
||||
- [x] **Idempotent (E3)**: copierea operatiei in `obs` e DERIVE-ON-EMPTY (doar cand `obs` e gol)
|
||||
ca re-importul/re-editarea sa NU dubleze textul ("Schimb ulei; Schimb ulei"). Test dedicat
|
||||
anti-dublu-concat.
|
||||
- [x] **Cuplaj preview-import**: `obs` se adauga in `EDIT_FIELDS` (`import_router.py:261`); `_merge_override`
|
||||
il propaga (obs e free-text, cade pe ramura ne-canonicalizata — fara strip "0", doar trim).
|
||||
- **Verificare E2E**: `POST /trimitere/{id}/corecteaza` cu `obs` -> persistat -> vizibil in detaliu;
|
||||
optional proba live RAR ca `obs` apare in FINALIZATA.
|
||||
|
||||
### US-006: Backend — prestatii multi-cod (lista) la editare/corectie
|
||||
**Ca** operator **vreau** sa adaug/sterg mai multe coduri RAR pe o trimitere **pentru ca** o
|
||||
comanda poate avea mai multe prestatii, asa cum accepta RAR.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/web/routes.py` (`/corecteaza`, `/repune` — **rescrie logica single-`prestatii[0]`**
|
||||
de azi: `cod_prestatie_curent` la `routes.py:977-982` + injectia la `1146-1164`/`1288-1324`
|
||||
presupun UN cod; multi-select cere pre-fill din lista intreaga + scriere pe toti itemii),
|
||||
`app/api/v1/import_router.py` (`/editeaza`, idem), `app/mapping.py` (NEATINS — deja accepta lista),
|
||||
`app/validation.py` (fiecare cod in nomenclator), `tests/test_web_corectie*.py`,
|
||||
`tests/test_mapping*.py` (~6 fisiere). Nota: `mapping.py` e neatins, dar call-site-urile din
|
||||
handler-e cer un rewrite real (nu "fara schimbare de logica").
|
||||
- **Test intai (RED)**: `tests/test_web_corectie_prestatii.py` — `test_mai_multe_coduri_acceptate`, `test_cod_invalid_respins`, `test_lista_goala_needs_mapping`, `test_idempotency_recalculat`, `test_odometru_initial_conditionat_R_ODO`
|
||||
- **Acceptance criteria**:
|
||||
- [x] Handler-ele de editare accepta o LISTA de `cod_prestatie`, inlocuind selectul unic. **NU
|
||||
reconstrui lista cu itemi goi**: handler-ele de azi injecteaza codul DOAR in `prestatii[0]`
|
||||
(`routes.py:1146-1164`, `1288-1324`) — multi-select le rescrie ca: pastreaza itemii existenti
|
||||
cu `cod_op_service`/`denumire` (invariant D7) si seteaza/adauga `cod_prestatie` pe ei.
|
||||
`idPrezentare:null` se adauga in `payload.py` la construirea payload-ului, NU in itemul intern.
|
||||
**E1 (CRITIC)**: `/repune` face azi `p0.pop("cod_op_service", None)` la `routes.py:1326` —
|
||||
ELIMINA acel `pop`: cand se seteaza un cod direct, `cod_op_service`/`denumire` RAMAN pe item
|
||||
(altfel rupe D7 si US-009). **Test de regresie obligatoriu** (IRON RULE): op_service
|
||||
supravietuieste unui /repune cu cod.
|
||||
- [x] **Pereche operatie<->cod definita**: cand exista operatii (`cod_op_service`), fiecare cod-chip
|
||||
se ataseaza unei operatii (1 operatie -> 1 cod, ca azi, dar acum N operatii -> N coduri);
|
||||
cand NU exista operatie (cod direct, ex. corectie pura), chip-urile sunt coduri libere intr-o
|
||||
lista fara `op_service`. Aceasta pereche e ce consuma US-009 (salvare mapare op->cod).
|
||||
- [x] Fiecare cod e validat fata de nomenclator (`valid_codes`); cod necunoscut -> respins cu
|
||||
mesaj (NU se trimite raw — invariant ORA-12899 din CLAUDE.md/contract).
|
||||
- [x] Lista goala de coduri -> ramane `needs_mapping` (nu se trimite fara cod).
|
||||
- [x] **Coduri duplicate** -> dedupare **PER-ITEM, nu "dupa cod"** (E4): doua operatii distincte
|
||||
pot mapa legitim la acelasi cod RAR; deduparea naiva dupa cod ar sterge o operatie reala si
|
||||
ar distruge contextul op->cod cerut de US-009. Dedup = acelasi (op, cod) de 2x, nu acelasi cod.
|
||||
- [x] Recalcul idempotenta dupa editare (mecanism existent), cu prinderea coliziunii ca azi.
|
||||
- [x] Se pastreaza regula `odometruInitial` obligatoriu cand lista contine `R-ODO`/`I-ODO`
|
||||
(contract §payload) — validare existenta, doar verificata pe lista.
|
||||
- **Verificare E2E**: `POST /corecteaza` cu 2 coduri valide -> `queued` cu `prestatii` de lungime 2;
|
||||
cu un cod invalid -> respins; optional live RAR cu 2 prestatii -> FINALIZATA.
|
||||
|
||||
### US-007: UI — formular editare slim (VIN unic, Observatii, chips prestatii)
|
||||
**Ca** operator **vreau** formularul de editare in design slim cu chips de prestatii **pentru ca**
|
||||
e compact si imi arata clar codurile RAR si observatiile, ca in mockup.
|
||||
|
||||
- **Depinde de**: US-002, US-005, US-006
|
||||
- **Fisiere**: `app/web/templates/_form_editare.html`, `app/web/templates/_macros.html`,
|
||||
`app/web/templates/_trimitere_detaliu.html`, `app/web/templates/_editare_preview_modal.html`,
|
||||
`tests/test_web_preview_edit.py`, `tests/test_web_detaliu*.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_form_editare_slim.py` — `test_un_singur_vin`, `test_camp_observatii_prezent`, `test_chips_multi_select_prestatii`, `test_adauga_sterge_chip`, `test_form_slim_in_ambele_modale`
|
||||
- **Acceptance criteria**:
|
||||
- [x] Formularul foloseste varianta slim de camp (US-002): VIN, Data prestatiei, Nr. inmatriculare,
|
||||
Observatii (textarea), prestatii (chips), Odometru — un SINGUR camp VIN (fara "Confirma VIN").
|
||||
- [x] Observatii = textarea liber, legat de `obs` (US-005).
|
||||
- [x] Prestatii = chips multi-select. **Binding op<->cod (E4)**: cand exista operatii
|
||||
(`cod_op_service`), UN picker PE operatie (eticheta op + chip-ul ei de cod), pastrand
|
||||
perechea per-item; lista plata de coduri libere DOAR pentru cazul fara operatie (corectie
|
||||
pura). Fiecare cod ca chip cu `×`; lista se trimite ca `cod_prestatie` multiplu (US-006).
|
||||
- [x] Acelasi `_form_editare.html` slujeste ambele modale (detaliu `/corecteaza` si preview
|
||||
`/editeaza`), fara duplicare; degradare fara JS rezonabila (chips ca lista, picker = select).
|
||||
- [x] **Require dinamic odometruInitial** (D10c): cand lista de chips contine `R-ODO` sau `I-ODO`,
|
||||
formularul DEZVALUIE si cere `odometru_initial` (contract §payload), previne 400 RAR si un
|
||||
drum `needs_data`. Cand niciun chip R-ODO/I-ODO -> campul ramane optional/ascuns.
|
||||
- [x] **Editare keyboard-first** (D10d): in picker, Enter adauga chip-ul selectat; sageti
|
||||
navigheaza optiunile; Esc inchide modalul; focus-ul revine logic dupa adaugare/stergere.
|
||||
- [x] Stilizare fidela mockup-ului pe toate temele; tinte 44px pe mobil; a11y (label-uri, aria,
|
||||
anunt de chip adaugat/sters pentru screen-reader).
|
||||
- [x] **HTMX server-driven PRIMARY (E6)**: chips add/remove via `hx-post` care re-randeaza
|
||||
partial-ul chips+form; reveal-ul conditional `odometruInitial` rezulta GRATIS din re-randarea
|
||||
server (server computeaza din lista de chips, fara ramura JS); navigare tastatura =
|
||||
`<select>`/`<datalist>` nativ. JS custom DOAR ca progressive enhancement (snappiness), nu
|
||||
calea principala. Elimina path-ul dublu JS/no-JS pe care formularea anterioara il cerea.
|
||||
- [x] **Referinta vizuala (criteriu din mockup)**: `docs/mockups/prd-5.15-mockups.html` defineste
|
||||
aspectul-tinta — VIN unic (FARA al doilea camp "Confirma VIN" din mockup-ul landing); Observatii
|
||||
ca textarea slim; picker PE operatie cu DOUA stari vizuale: (a) operatie mapata = chip cod cu `×`
|
||||
+ "+ alt cod" + link "salveaza regula op->cod" (US-009); (b) operatie ne-mapata = picker galben
|
||||
"alege cod RAR" cu eticheta "lipsa cod". OdometruInitial: ascuns implicit (doar hint discret
|
||||
"se cere doar pentru R-ODO/I-ODO") si DEZVALUIT cu bordura-stanga galbena + label "necesar pentru
|
||||
R-ODO" cand lista de chips contine R-ODO/I-ODO.
|
||||
- **Verificare E2E**: browser — editare trimitere needs_data: schimb VIN + scriu Observatii + adaug
|
||||
2 coduri RAR (chips, cu tastatura) + adaug R-ODO (apare odometruInitial) + sterg un chip -> salvare
|
||||
-> persistat; identic in preview import.
|
||||
|
||||
### US-008: Teste de regresie + E2E final pe cele 4 teme
|
||||
**Ca** dezvoltator **vreau** acoperire si o trecere E2E completa **pentru ca** redesign-ul atinge
|
||||
fisiere fierbinti (base.html) si nu vreau regresii pe teme/liste/formular.
|
||||
|
||||
- **Depinde de**: US-003, US-004, US-007
|
||||
- **Fisiere**: `tests/test_web_responsive.py`, `tests/test_tema.py`, `tests/test_web_submissions.py`
|
||||
(~3 fisiere)
|
||||
- **Test intai (RED)**: completare scenarii lipsa (componente noi pe TOATE temele; slim list desktop+mobil)
|
||||
- **Acceptance criteria**:
|
||||
- [x] `pytest -q -m "not live"` verde (fara regresii fata de baseline).
|
||||
- [x] **Test de tema robust, nu esantion**: un test parametrizat verifica fiecare token critic
|
||||
(`--card2`, `--line2`, `--accent`, `--ok`, `--err`) e DEFINIT in TOATE cele 7+1 stari
|
||||
(light/dark/petrol/grafit/cobalt/cupru/hartie/Auto). Ancorare pe SENTINEL CSS (nu felii
|
||||
fixe `[idx:idx+N]`) — vezi regresia false-green din ROADMAP 5.13.
|
||||
- [ ] E2E Playwright pe 390/820/1280, pe un dark (grafit) + un light (hartie) + petrol (verificare
|
||||
ca temele vechi nu au regresat): strip sanatate, dashboard contoare, lista slim cu
|
||||
filtre/paginare/bulk, formular slim cu chips, fara overflow orizontal.
|
||||
|
||||
### US-009: Salvare mapare din chip (compounding cu fluxul de mapare)
|
||||
**Ca** operator **vreau** ca atunci cand adaug un cod RAR la o operatie sa-l pot salva ca regula
|
||||
**pentru ca** data viitoare operatia sa se auto-rezolve, fara sa re-mapez manual.
|
||||
|
||||
- **Depinde de**: US-006, US-007
|
||||
- **Fisiere**: `app/web/templates/_form_editare.html`, `app/web/routes.py` (reuse `save_mapping` +
|
||||
`reresolve_account` — fara logica noua), `tests/test_web_mapare_din_chip.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_mapare_din_chip.py` — `test_salveaza_regula_din_chip`, `test_reresolve_deblocheaza_frate`, `test_optional_nu_forteaza`
|
||||
- **Acceptance criteria**:
|
||||
- [x] Cand operatia (`op_service`) e cunoscuta si userul adauga un cod RAR prin chip, apare optiunea
|
||||
"salveaza ca regula op->cod"; la confirmare reuse EXACT `save_mapping` + `reresolve_account`
|
||||
(acelasi mecanism ca maparea inline din 5.7), scoped pe cont + CSRF.
|
||||
- [x] Re-rezolvarea deblocheaza si alte submission-uri `needs_mapping` cu aceeasi operatie (pe `batch_id`).
|
||||
- [x] Optional: daca userul nu vrea sa salveze, editarea ramane one-off (fara regula). Se compune
|
||||
cu 5.14 (auto-maparea umple, salvarea din chip ramane fallback-ul uman).
|
||||
- **Verificare E2E**: adaug cod la operatie nemapata + salveaza regula -> al doilea rand cu aceeasi
|
||||
operatie se rezolva automat.
|
||||
|
||||
### US-010: Bulk-fix din lista (selectie multipla -> actiune unica)
|
||||
**Ca** operator **vreau** sa corectez mai multe randuri blocate dintr-o data **pentru ca** la 2-20
|
||||
de corectat/zi nu vreau sa intru in fiecare individual.
|
||||
|
||||
- **Depinde de**: US-004, US-006
|
||||
- **Fisiere**: `app/web/templates/_submissions.html`, `app/web/routes.py` (reuse infra bulk
|
||||
existenta din `_submissions` + `submissions_admin`), `tests/test_web_bulk_fix.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_bulk_fix.py` — `test_bulk_remapeaza_selectie`, `test_bulk_doar_blocate`, `test_bulk_scoped_cont`
|
||||
- **Acceptance criteria**:
|
||||
- [x] Pe randurile blocate (checkbox existent pe `gestionabil`), o actiune bulk noua: aplica un cod
|
||||
RAR / o remapare la toata selectia intr-o singura cerere (reuse forma `#bulk-trimiteri`).
|
||||
- [x] Scoped pe cont (404-before-409 ca la bulk-delete); doar randuri blocate eligibile.
|
||||
- [x] Fiecare rand re-validat + idempotenta recalculata individual (un cod invalid pe un rand nu
|
||||
pica tot lotul — sumar "N reusite, M esuate" ca la salvarea mapcoloane D#12).
|
||||
- **Verificare E2E**: selectez 3 randuri needs_mapping + aplic un cod -> toate 3 -> `queued`.
|
||||
- **Verificare E2E**: rulare completa documentata in Raportul VERIFY.
|
||||
|
||||
### US-011: Securitate — account-scope pe GET-urile de listare (interleave, E8)
|
||||
**Ca** operator **vreau** ca listarile sa-mi arate DOAR trimiterile contului meu **pentru ca**
|
||||
azi GET-urile de listare sunt globale + neprotejate (scurgere VIN/PII cross-cont, notata in CLAUDE.md).
|
||||
|
||||
- **Depinde de**: — (backend pur, independent de UI; ruleaza in paralel cu valurile de design)
|
||||
- **Fisiere**: `app/web/routes.py` (GET-urile de listare trimiteri), `app/api/v1/router.py`
|
||||
(GET-urile API de listare daca sunt globale), `app/auth.py` (refolosire scope existent),
|
||||
`tests/test_web_scope.py`, `tests/test_api_scope.py` (~5 fisiere)
|
||||
- **Test intai (RED)**: `test_get_listare_scoped_cont` — un cont NU vede randuri ale altui cont;
|
||||
`test_get_listare_neautentificat_401`; `test_get_detaliu_scoped` (404-before-leak pe id strain).
|
||||
- **Acceptance criteria**:
|
||||
- [x] GET-urile de listare (trimiteri + orice listare globala) devin account-scoped, refolosind
|
||||
mecanismul de scope existent (ca POST-urile + bulk-delete: 404-before-409 pe id strain).
|
||||
- [x] Un cont nu poate enumera/citi VIN/PII al altui cont prin listare sau detaliu.
|
||||
- [x] Enforcement aliniat cu `AUTOPASS_REQUIRE_API_KEY` (dev vs prod), fara a rupe contul id=1
|
||||
implicit in dev.
|
||||
- [x] Actualizeaza nota din CLAUDE.md ("GET-urile de listare ... de remediat") cand e inchis.
|
||||
- **Verificare E2E**: doua conturi cu trimiteri; contul A nu vede niciun rand al contului B in
|
||||
listare, filtre, paginare sau detaliu.
|
||||
|
||||
### US-012: Analytics device-mix (validare premisa mobil, in-PR)
|
||||
**Ca** owner **vreau** sa stiu raportul desktop/mobil al operatorilor **pentru ca** sa decid daca
|
||||
rafinarile mobil (390px) viitoare merita efortul (premisa nevalidata din TODOS 5.13/CEO-F1).
|
||||
|
||||
- **Depinde de**: — (instrumentare backend, independenta de UI)
|
||||
- **Fisiere**: `app/web/routes.py` (sau middleware existent), `app/schema.sql` SAU `app_events`
|
||||
(reuse tabela de evenimente existenta — fara coloana noua daca `app_events` poarta semnalul),
|
||||
`tests/test_device_mix.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `test_device_mix_inregistrat`, `test_device_mix_fara_pii`.
|
||||
- **Acceptance criteria**:
|
||||
- [x] La acces dashboard, clasifica grosier viewport/UA in desktop/mobil si inregistreaza in
|
||||
`app_events` (semnal agregat, FARA PII suplimentar). Reuse tabela existenta — fara migrare
|
||||
daca `app_events` poarta semnalul.
|
||||
- [x] Un mod simplu de citire a raportului (query/admin), suficient pentru a decide investitia mobil.
|
||||
- [x] Zero PII nou; aliniat retentiei `app_events` existente.
|
||||
- **Verificare E2E**: acces dashboard de pe doua viewport-uri -> doua evenimente clasificate corect.
|
||||
|
||||
## 4. Riscuri
|
||||
|
||||
- **base.html fisier fierbinte**: US-001/US-002 il ating amandoua + US-003/004/007 il citesc.
|
||||
Serializeaza pe valuri (un singur autor pe val pe base.html), ca la 5.12/5.13.
|
||||
- **Migrare teme legacy**: useri cu `localStorage.theme` = light/dark/petrol. Mitigare: maparea
|
||||
grafioasa din US-001 (light->hartie, dark->grafit, petrol->grafit) + test dedicat.
|
||||
- **Restyle lista = pierdere de functii**: filtre/paginare/bulk pot fi sparte de schimbarea de
|
||||
markup. Mitigare: US-004 are AC explicite pentru pastrarea lor + teste lock.
|
||||
- **Idempotenta la prestatii multiple**: schimbarea listei schimba cheia canonica. Mitigare:
|
||||
refolosim mecanismul existent de recalcul + prindere coliziune (3.5/5.10), zero logica noua.
|
||||
- **Densitate vizuala pe mobil**: randul slim cu 2 linii + pill poate aglomera. Mitigare: tinte
|
||||
44px + verificare 390px in US-004/008.
|
||||
- **Premisa mobil nevalidata** (din TODOS 5.13, CEO F1): valoarea slim/compact pe mobil presupune
|
||||
utilizare reala pe mobil. Daca device-mix-ul e ~95% desktop, partea responsive e efort irosit.
|
||||
Mitigare: nu blocheaza (designul e bun si pe desktop), dar confirma analytics inainte de a investi
|
||||
in rafinari mobil viitoare.
|
||||
- **7 teme = suprafata de test/intretinere** pe fisierul cel mai fierbinte: fiecare componenta noua
|
||||
trebuie corecta in 7+1 stari. Istoricul (5.13) arata ca testele de tema au dat false-green o data.
|
||||
Mitigare: US-008 cere test parametrizat ancorat pe SENTINEL (nu felii fixe); deduparea
|
||||
grafit~dark / hartie~light ramana optiune de simplificare (reziduala, non-blocanta).
|
||||
- **Secventiere cu 5.14** (D9): 5.15 defineste forma listei `prestatii`; daca 5.14 (mapare LLM)
|
||||
porneste in paralel, sincronizeaza forma listei. Mitigare: 5.15 INAINTE de 5.14.
|
||||
|
||||
## 5. Intrebari deschise
|
||||
|
||||
> Toate intrebarile au fost REZOLVATE cu userul (vezi D1-D5 §1). Pastrate aici ca istoric al deciziei.
|
||||
|
||||
- **I1 — contor Trimise** [REZOLVAT]: arata all-time + luna asta + azi (D4). `_status_counts` extins.
|
||||
- **I2 — teme** [REZOLVAT]: aditiv — light/dark/petrol + Auto + grafit/cobalt/cupru/hartie (D2).
|
||||
- **I3 — stocare obs** [REZOLVAT]: in `payload_json`, fara coloana noua (D5).
|
||||
- **I4 — operatii la import -> obs** [REZOLVAT]: concatenam denumirea operatiei in `obs` cand
|
||||
fisierul nu are coloana Observatii (D5).
|
||||
- Reziduale minore (de decis la executie, non-blocante): formatul exact de concatenare a denumirilor
|
||||
in `obs`; rezolvarea "Auto" la light vs hartie; eventuala deduplicare grafit~dark / hartie~light
|
||||
in eticheta selectorului.
|
||||
|
||||
## 6. Valuri de executie (graful de dependente)
|
||||
|
||||
```
|
||||
Val 0: [US-011] authz GET-listari (backend pur; ruleaza in paralel cu orice val) ||
|
||||
Val 1: [US-001] base.html teme + tokeni (autor unic pe base.html)
|
||||
Val 2: [US-002] base.html componente (dupa US-001, autor unic pe base.html)
|
||||
Val 3: [US-003] [US-004] dashboard + strip sanatate + lista (consuma US-002; disjuncte) ||
|
||||
[US-005] -> [US-006] backend obs APOI prestatii — SECVENTIAL (E5): ambele rescriu
|
||||
ACEEASI functie post_corecteaza (routes.py:1120-1262), autor unic
|
||||
Val 4: [US-007] formular slim cu chips (dupa US-002+US-005+US-006)
|
||||
Val 5: [US-009] [US-010] salvare mapare din chip || bulk-fix (dupa US-006/007 resp. US-004)
|
||||
— disjuncte la nivel de template (_form_editare vs _submissions)
|
||||
Val 6: [US-008] regresie + E2E final (dupa toate)
|
||||
```
|
||||
|
||||
> **Regula autor-unic extinsa (E5)**: pe langa base.html, `routes.py` are autor unic in Val 3:
|
||||
> US-005 INAINTE de US-006 (ambele in `post_corecteaza`). US-009/US-010 in Val 5 sunt disjuncte
|
||||
> la nivel de template; adauga rute noi separate in routes.py (regiuni diferite, mergeabile).
|
||||
|
||||
> Secventiere fata de alte PRD-uri (D9): **5.15 INAINTE de 5.14** (mapare LLM) — 5.15 fixeaza forma
|
||||
> listei `prestatii` si UX-ul de confirmare; 5.14 umple codurile peste aceeasi forma.
|
||||
|
||||
---
|
||||
|
||||
## Raport VERIFY
|
||||
|
||||
Verificator independent (context curat, subagent Sonnet) — 2026-06-28. **VERDICT: PASS** (12/12 stories),
|
||||
cu 1 FAIL documentar remediat de lead + 1 OPEN limitat de mediu.
|
||||
|
||||
- **Suita completa**: `python3 -m pytest -q -m "not live"` → **1230 passed, 1 deselected, 0 failed** (118s).
|
||||
Baseline initial 992 → +238 teste, zero regresii.
|
||||
- **AC per story (US-001..US-012)**: toate PASS cu dovezi (fisier:linie + test care le acopera).
|
||||
Puncte verificate explicit: 7+1 teme cu `--card2`/`--line2` in toate (US-001, DRY `THEMES`);
|
||||
componente slim doar cu tokeni, zero hex (US-002, ancorat pe `SENTINEL-COMPONENTE-SLIM`);
|
||||
strip sanatate D6 + 3 contoare + `sent_today`/`sent_month` bucketate timp local RO `+3 hours` (US-003, E7);
|
||||
lista slim cu filtre/paginare/bulk pastrate (US-004); `obs` editabil + EXCLUS din cheia idempotenta
|
||||
(`idempotency.py:98`) + concat derive-on-empty anti-dublu (US-005, D8/E3); prestatii multi-cod via
|
||||
`getlist` + **E1 IRON RULE** (`cod_op_service` supravietuieste `/repune` — test dedicat) + dedup per-item
|
||||
(US-006, E4); form slim VIN unic + picker chips pe operatie + reveal odo server-driven + select vechi
|
||||
redundant ELIMINAT (US-007/cleanup B); test tema parametrizat 5 tokeni x 7 teme ancorat pe selectori
|
||||
`[data-theme]` (US-008, anti false-green); salvare mapare din chip reuse `save_mapping`+`reresolve_account`
|
||||
(US-009); bulk-fix sumar "N reusite/M esuate" scoped (US-010); account-scope GET-listari 404-before-leak
|
||||
(US-011); device-mix fara PII reuse `app_events` (US-012).
|
||||
- **Fidelitate mockup** (`docs/mockups/prd-5.15-mockups.html`, cod-level): D6 strip, contoare D4,
|
||||
picker E4 cu 2 stari (mapata=chip+×+salveaza / nemapata=select galben "lipsa cod"), reveal odo
|
||||
border-left warn — toate conforme; toate culorile prin `var(--token)`, fara hex.
|
||||
- **Regresia de aur**: testele `POST /v1/prezentari` + worker + import→commit raman verzi in suita;
|
||||
E1 confirmat cu test. Live RAR real (`FINALIZATA`) = opt-in, indisponibil fara creds in sandbox (documentat).
|
||||
|
||||
**FAIL 1 (remediat de lead)**: nota CLAUDE.md "GET-urile de listare globale + neprotejate (de remediat)"
|
||||
nu fusese actualizata (teammates instruiti sa NU atinga CLAUDE.md). **Remediat**: `CLAUDE.md:70` actualizat
|
||||
sa reflecte scope-ul implementat de US-011.
|
||||
|
||||
**OPEN (mediu)**: E2E Playwright pe 390/820/1280 (grafit/hartie/petrol) — browserul MCP a returnat
|
||||
"already in use" in sandbox (ca la livrabilele anterioare). Serverul porneste OK (`/healthz` ok),
|
||||
ACs acoperite functional de pytest (`test_web_responsive.py`). Recomandat: rulat de operator cu browser real.
|
||||
|
||||
---
|
||||
|
||||
## GSTACK REVIEW REPORT
|
||||
|
||||
Review: `/plan-ceo-review` — 2026-06-28. Mod: **SELECTIVE EXPANSION**. Model: claude (opus).
|
||||
Abordare aleasa de user: tot PRD-ul (8 stories) + 4 extinderi acceptate -> **10 stories**.
|
||||
|
||||
| Pasaj | Status | Constatari materiale |
|
||||
|-------|--------|----------------------|
|
||||
| Audit sistem | OK | base.html cel mai fierbinte fisier (31x/30z); 5.15 = a 5-a iteratie pe acelasi UI (smell recurent); 5.14 in flight pe acelasi seam |
|
||||
| S1 Arhitectura | OK | Fara componente noi; fara migrare; rollback = revert template. Concentrare de risc pe base.html, nu coupling |
|
||||
| S2 Eroare/Rescue | 2 GAP | (a) coduri duplicate in chips nedefinit -> US-006 dedupe; (b) cod necunoscut: invariant ORA-12899 pastrat |
|
||||
| S4 Edge cases | 1 GAP HIGH | R-ODO/I-ODO cere odometruInitial; formularul nu il forta -> US-007 require dinamic (D10c) |
|
||||
| S2/Idempotenta | 1 FIX | `obs` EXCLUS din cheie (`idempotency.py:98`) -> AC US-005 corectat (D8); `prestatii` in cheie -> re-cheiere OK |
|
||||
| S6 Test | 1 GAP | 7 teme x componente pe fisier fierbinte; "esantion" prea lax -> US-008 test parametrizat ancorat pe SENTINEL |
|
||||
| S8/S11 Trust | 1 HIGH | carduri-contor ascundeau sanatatea -> strip mereu-vizibil deasupra contoarelor (D6) |
|
||||
| S9 Deploy | OK | Fara migrare; doar sent_today/sent_month (scoped). Rollback ieftin |
|
||||
| S10 Trajectorie | 1 DECIZIE | secventiere 5.15 inainte de 5.14 (D9) |
|
||||
| S11 Design/UX | OK + 4 EXT | strip trust; extinderi: salvare mapare din chip, bulk-fix, require dinamic odo, keyboard-first |
|
||||
|
||||
**Decizii incorporate (D6-D10):** strip sanatate mereu-vizibil (D6); operatie ramane in op_service +
|
||||
copiata in obs (D7); obs exclus din idempotenta, AC corectat (D8); 5.15 inainte de 5.14 (D9); cele 4
|
||||
extinderi acceptate (D10) -> US-007 imbogatit + US-009 (salvare mapare din chip) + US-010 (bulk-fix).
|
||||
|
||||
**Risc rezidual notat (non-blocant):** premisa "utilizare mobil reala" nevalidata (TODOS 5.13 F1);
|
||||
7 teme = suprafata de test pe fisier fierbinte (deduparea grafit~dark/hartie~light ramana optiune).
|
||||
|
||||
**Spec-review loop (reviewer independent, context curat) — scor 7/10, verdict ISSUES -> remediat:**
|
||||
- #1 HIGH (contradictie): US-006 spunea "reconstruieste prestatii ca itemi goi `{cod_prestatie}`",
|
||||
ceea ce ar fi sters `cod_op_service`/`denumire` -> rupea D7 si US-009. **Remediat**: US-006 pastreaza
|
||||
itemii existenti, seteaza doar `cod_prestatie`; pereche operatie<->cod definita; `idPrezentare` se
|
||||
adauga in `payload.py`, nu in itemul intern.
|
||||
- #2 MEDIUM: `sent_today`/`sent_month` nu aveau sursa de timp (nu exista `sent_at`). **Remediat**:
|
||||
US-003 foloseste `status='sent' AND date(updated_at)` cu justificare (randul `sent` nu mai e scris
|
||||
pana la purge la +90z) -> fara migrare.
|
||||
- #3 MEDIUM: US-006 subestima rewrite-ul handler-elor (logica single-`prestatii[0]`). **Remediat**:
|
||||
Fisierele US-006 numesc liniile exacte de rescris.
|
||||
- #5/#6 LOW: suprafata JS reala (US-007) + tinta de click "De corectat" (US-003). **Remediat** (note adaugate).
|
||||
- #4 LOW (scope): US-009/US-010 sunt adiacente FUNCTIONALE (din SELECTIVE EXPANSION), dincolo de
|
||||
obiectivul pur de propagare design. **Acceptat constient** (alegerea userului); ramane optiunea de
|
||||
a le scoate intr-un PRD separat daca propagarea design e ce e urgent.
|
||||
|
||||
### /plan-eng-review — 2026-06-28 (model claude/opus; outside-voice = Claude subagent, Codex usage-limit)
|
||||
|
||||
| Review | Trigger | Why | Runs | Status | Findings |
|
||||
|--------|---------|-----|------|--------|----------|
|
||||
| CEO Review | `/plan-ceo-review` | Scope & strategy | 1 | issues_open->remediat | 10 stories, 4 ext acceptate, spec-review remediat |
|
||||
| Eng Review | `/plan-eng-review` | Architecture & tests (required) | 1 | issues_open->remediat | 7 issues (2 HIGH), 1 regresie IRON-RULE, +2 stories noi |
|
||||
| Outside Voice | Claude subagent | Independent 2nd opinion | 1 | issues_found | 10 findings; 2 HIGH absorbite, restul foldate |
|
||||
|
||||
**Step 0 scope:** acceptat ca-atare (10 stories). Gate de complexitate = breadth, nu depth (zero clase/servicii noi). User a confirmat pastrarea US-009/US-010.
|
||||
|
||||
**Constatari eng-review (toate confirmate cu userul si foldate in AC):**
|
||||
- **E1 (ARCH, HIGH, conf 9/10)** `routes.py:1326` `/repune` face `p0.pop("cod_op_service")` — sterge operatia, rupe D7+US-009. US-006: elimina pop + test regresie (IRON RULE).
|
||||
- **E2 (Code-quality, conf 9/10)** config teme duplicat ~7 locuri pe base.html (anti-FOUC + 5 literali). US-001: o singura structura `THEMES`, restul derivat.
|
||||
- **E3 (Test, conf 7/10)** obs concat la import poate dubla textul la re-import. US-005: derive-on-empty + test anti-dublu.
|
||||
- **E7 (Perf/Correctness, conf 8/10 — outside-voice)** `date(updated_at)` UTC numara gresit `azi`/`luna` peste granita local RO. US-003: bucketare timp local + test granita.
|
||||
|
||||
**Outside-voice (Claude subagent) — material absorbit:**
|
||||
- **OV/E4 (HIGH, conf 9/10)** chip-uri lista plata fara binding op<->cod -> rupe US-009; dedup-dupa-cod sterge operatie legala. US-006/007: picker PE operatie cand exista op; flat doar fara op; dedup per-item.
|
||||
- **OV/E5 (HIGH, conf 9/10)** Val 3 conflict same-function: US-005+US-006 rescriu `post_corecteaza`. §6: serializare US-005 -> US-006 pe routes.py.
|
||||
- **OV/E6 (MED, conf 8/10)** US-007 supraestimeaza JS custom intr-un app HTMX. US-007: hx-post server-driven primary; reveal odo gratis; select/datalist nativ.
|
||||
- **OV/E8 (MED securitate, conf 7/10)** GET-uri de listare globale neprotejate (scurgere VIN/PII cross-cont, CLAUDE.md). User a ales INTERLEAVE -> **US-011** (account-scope + teste).
|
||||
- **OV minore foldate:** reconcile drift pe `updated_at` (caveat US-003); cost poll non-sargabil (notat, non-blocant); cuplaj `EDIT_FIELDS` pentru obs preview (US-005 AC).
|
||||
|
||||
**TODO (decizie user):** premisa mobil nevalidata -> user a ales BUILD-IN-PR -> **US-012** (analytics device-mix, fara PII, reuse `app_events`).
|
||||
|
||||
**Scope actualizat:** 10 -> **12 stories** (+US-011 authz, +US-012 analytics). Fara migrare de schema. Outside-voice a confirmat "no migration" TRUE.
|
||||
|
||||
**Failure modes — gap critic:** niciun gap critic ramas silent. Cel mai aproape: E1 (regresie tacuta op_service pe /repune) — acum acoperit de test obligatoriu. E7 (off-by-a-day tacut) — acum cu test de granita.
|
||||
|
||||
**VERDICT:** CEO + ENG CLEARED — gata de executie. 12 stories. Outside-voice absorbit (2 HIGH foldate). Fara migrare de schema.
|
||||
|
||||
NO UNRESOLVED DECISIONS
|
||||
1064
docs/prd/prd-5.16-tipografie-uniforma-bugfix-editare.md
Normal file
1064
docs/prd/prd-5.16-tipografie-uniforma-bugfix-editare.md
Normal file
File diff suppressed because it is too large
Load Diff
960
docs/prd/prd-5.17-tipuri-cont-planuri-trial.md
Normal file
960
docs/prd/prd-5.17-tipuri-cont-planuri-trial.md
Normal file
@@ -0,0 +1,960 @@
|
||||
<!-- /autoplan restore point: /home/claude/.gstack/projects/romfast-rar-autopass/docs-prd-5.16-5.17-design-tiers-autoplan-restore-20260628-212453.md -->
|
||||
# PRD 5.17 — Tipuri de cont (planuri) + trial Pro 30 zile + enforcement
|
||||
|
||||
**Stare**: draft
|
||||
|
||||
> Proces complet: `docs/ROADMAP.md` §5. Contract RAR (sursa de adevar): `docs/api-rar-contract.md`.
|
||||
> Landing comercial cu planurile: `app/web/templates/landing.html` (sectiunea PRICING).
|
||||
> Lifecycle cont existent: `app/accounts.py`, `app/schema.sql` (tabela `accounts`, coloana `status`).
|
||||
> Signup: `app/web/auth_routes.py` (`signup_post`, butoanele landing trimit `data-plan`).
|
||||
> Starea trece: `draft -> aprobat -> in-executie -> verify-pass -> inchis`.
|
||||
|
||||
## 1. Introducere
|
||||
|
||||
Landing-ul comercial promite patru planuri — **Gratuit**, **Standard (39 lei)**, **Pro (59 lei, cu
|
||||
API)**, **Premium (la cerere)** — si afirma ca **fiecare cont incepe cu acces gratuit 30 de zile** la
|
||||
un plan superior. In aplicatie insa **nu exista deloc conceptul de tip de cont**: tabela `accounts`
|
||||
are doar `status` (pending/active/blocked/archived/deleted) si `on_unmapped_error_default`. Nimic nu
|
||||
diferentiaza un cont gratuit de unul platit, nimic nu aplica limita de volum sau gate-ul de API, si nu
|
||||
exista niciun trial.
|
||||
|
||||
In plus, userul a decis doua corectii fata de landing-ul actual:
|
||||
1. Trial-ul de 30 de zile e pe **Pro**, NU pe Premium (landing-ul scrie azi "Premium gratuit 30 de
|
||||
zile" — gresit; trebuie "Pro 30 de zile").
|
||||
2. Limita planului **Gratuit** scade de la **100** la **60 de prestatii/luna** — actualizata si in
|
||||
landing si in aplicatie.
|
||||
|
||||
5.17 introduce modelul de tipuri de cont, trial-ul Pro de 30 de zile, **enforcement DUR** al
|
||||
diferentelor (volum lunar + acces API), si downgrade automat la expirarea trial-ului. NU include
|
||||
integrare de plata (nu exista inca sistem de facturare) — alocarea planului platit ramane manuala
|
||||
(admin), iar trial-ul porneste automat la creare cont.
|
||||
|
||||
## 2. Obiective
|
||||
|
||||
### Obiectiv principal
|
||||
Aplicatia sa sustina real diferentele dintre planuri pe care landing-ul le promite: cont nou →
|
||||
trial Pro 30 zile → la expirare downgrade pe Gratuit (60/luna, fara API), cu enforcement efectiv.
|
||||
|
||||
### Obiective secundare
|
||||
- Sursa unica de adevar pentru definitia planurilor (limite + capabilitati), consumata de backend si UI.
|
||||
- Mesaje oneste cand un cont atinge limita sau cere o capabilitate neinclusa (3 niveluri, ca 5.4).
|
||||
- Vizibilitate in dashboard: planul curent + zile ramase din trial + consum lunar.
|
||||
|
||||
### Metrici de succes
|
||||
- Un cont Gratuit care depaseste 60 prestatii/luna primeste un raspuns clar de respingere (API + web),
|
||||
iar contoarele lunare se reseteaza corect la inceput de luna (timp local RO).
|
||||
- Un cont fara plan Pro+ primeste 403 onest pe `/v1/*` de import API.
|
||||
- Un cont nou are trial Pro activ; dupa 30 zile (sau setand `trial_until` in trecut in test) trece
|
||||
automat pe Gratuit, cu enforcement-ul aferent.
|
||||
- Landing + app afiseaza coerent "60 prestatii/luna" si "Pro gratuit 30 de zile".
|
||||
|
||||
## 3. User Stories
|
||||
|
||||
> Database → backend → API → UI (ordinea dependentelor). Un singur autor pe `accounts.py`/`schema.sql`
|
||||
> in valul de model.
|
||||
|
||||
### US-001: Schema — `accounts.tier` + `trial_until` + definitia planurilor
|
||||
**Ca** sistem **vreau** sa stiu planul fiecarui cont si pana cand e in trial **pentru ca** restul
|
||||
logicii depinde de asta.
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/schema.sql` (coloane noi + migrare defensiva), `app/accounts.py` (helperi),
|
||||
`app/plans.py` (NOU — definitia planurilor, sursa de adevar), `tests/test_accounts.py` /
|
||||
`tests/test_plans.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `test_migrare_tier_trial_defensiva`, `test_plan_definitii`,
|
||||
`test_cont_nou_trial_pro_30z`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] `accounts` capata (migrare aditiva defensiva, ca `email`/`status` in 5.5/5.12):
|
||||
`tier TEXT NOT NULL DEFAULT 'free' CHECK (tier IN ('free','standard','pro','premium'))`
|
||||
si `trial_until TEXT` (nullable; ISO datetime UTC sau NULL daca nu e in trial).
|
||||
- [ ] `app/plans.py` = SINGURA sursa de adevar: dict `PLANS` cu, per plan,
|
||||
`{label, monthly_limit, api_access, ...}`. Valori: `free` → `monthly_limit=60`, `api_access=False`;
|
||||
`standard` → `monthly_limit=None` (nelimitat), `api_access=False`; `pro` → `monthly_limit=None`,
|
||||
`api_access=True`; `premium` → `monthly_limit=None`, `api_access=True`. (Aliniat landing-ului,
|
||||
cu limita Gratuit 60.)
|
||||
- [ ] Helper `effective_tier(account)`: daca `trial_until` e in viitor → randeaza ca `pro`
|
||||
(trial); altfel `tier`. (Trial-ul = acces Pro temporar peste tier-ul de baza `free`.)
|
||||
- [ ] `create_account` seteaza `tier='free'` si `trial_until = now + 30 zile` (trial Pro automat la
|
||||
creare). Contul implicit id=1 (dev) e exceptat / setat coerent (nu blocheaza dev-ul).
|
||||
- [ ] Migrare idempotenta (re-rulabila); conturile legacy fara `tier` primesc `free` + fara trial
|
||||
(sau trial calculat din `created_at` — decizie la executie; implicit: legacy → free fara trial).
|
||||
- **Verificare E2E**: creez cont nou → `tier=free`, `trial_until ≈ now+30z`, `effective_tier=pro`.
|
||||
|
||||
### US-002: Numarator de consum lunar (prestatii/luna pe cont)
|
||||
**Ca** sistem **vreau** sa stiu cate prestatii a trimis un cont in luna curenta **pentru ca** limita
|
||||
Gratuit (60/luna) se aplica pe acest numar.
|
||||
|
||||
- **Depinde de**: US-001
|
||||
- **Fisiere**: `app/accounts.py` SAU `app/plans.py` (`monthly_usage(conn, account_id)`),
|
||||
`tests/test_plans.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `test_consum_lunar_numara_sent_si_queued`, `test_consum_lunar_timp_local_ro`,
|
||||
`test_consum_lunar_resetare_luna_noua`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] `monthly_usage(conn, account_id)` numara prestatiile contului in luna calendaristica curenta.
|
||||
**Definitia "prestatie consumata"** (de fixat la executie, propus): randuri `submissions` ale
|
||||
contului cu `status` in (`queued`,`sending`,`sent`) cu `created_at` in luna curenta — adica
|
||||
prestatiile ACCEPTATE in coada, nu cele respinse/blocate. (Justificare: limita e pe ce trimitem
|
||||
la RAR, nu pe incercari esuate.) Alternativ doar `sent` — de decis; implicit: acceptate-in-coada.
|
||||
- [ ] **Timp local RO** (ca E7 din 5.15): bucketarea lunii foloseste offset RO (`created_at,'+3 hours'`
|
||||
sau echivalent), nu UTC pur, ca prestatiile de la granita de luna sa cada corect. Test la granita.
|
||||
- [ ] Scoped strict pe cont (nu numara cross-account).
|
||||
- [ ] Fara coloana noua daca `submissions.created_at` ajunge (respecta non-goal migrare minima).
|
||||
- **Verificare E2E**: cont cu N trimiteri in luna → `monthly_usage == N`; luna urmatoare → reset la 0.
|
||||
|
||||
### US-003: Enforcement DUR — limita lunara Gratuit (60) pe ambele canale
|
||||
**Ca** owner **vreau** ca un cont Gratuit care depaseste 60 prestatii/luna sa fie oprit **pentru ca**
|
||||
asa sustinem diferenta de plan promisa.
|
||||
|
||||
- **Depinde de**: US-001, US-002
|
||||
- **Fisiere**: `app/api/v1/router.py` (`create_prezentari`), `app/api/v1/import_router.py`
|
||||
(commit import), `app/errors.py` (cod nou `PLAN_LIMITA_LUNARA`), `app/web/routes.py` (commit web),
|
||||
`tests/test_api_scope.py` / `tests/test_web_*` / `tests/test_plans.py` (~6 fisiere)
|
||||
- **Test intai (RED)**: `test_free_peste_60_respins_api`, `test_free_peste_60_respins_import_web`,
|
||||
`test_pro_si_trial_nelimitat`, `test_eroare_3_niveluri_plan_limita`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] La enqueue (API `POST /v1/prezentari` + commit import web + commit import API), daca
|
||||
`effective_tier` are `monthly_limit` si `monthly_usage + nr_cerut > monthly_limit` → cererea
|
||||
e respinsa (sau respinsa partial, la limita) cu eroare 3 niveluri (`app/errors.py`, cod
|
||||
`PLAN_LIMITA_LUNARA`: problema "Ai atins limita planului Gratuit (60/luna)", cauza, fix
|
||||
"Treci pe Standard/Pro sau astepti luna viitoare"). NU se face enqueue peste limita.
|
||||
- [ ] `standard`/`pro`/`premium` si conturile in **trial Pro** → fara limita de volum.
|
||||
- [ ] Comportament la cerere de lot care depaseste partial limita (ex. 50 folosite, vin 20):
|
||||
decizie la executie — implicit RESPINGERE clara a intregului lot cu mesaj cat mai e disponibil
|
||||
("mai poti trimite 10 luna asta"), NU enqueue partial tacut (evita surprize). De confirmat.
|
||||
- [ ] Enforcement aliniat cu `AUTOPASS_REQUIRE_API_KEY` (dev vs prod): in dev, contul id=1 nu e
|
||||
blocat artificial (trial/standard coerent), ca dogfooding-ul sa nu se loveasca de limita.
|
||||
- [ ] **Idempotenta neatinsa**: respingerea pe limita se face INAINTE de `build_key`/enqueue; un
|
||||
retry idempotent al unei prestatii deja acceptate nu consuma din nou cota.
|
||||
- **Verificare E2E**: cont free cu 60 trimise → a 61-a respinsa cu mesaj 3 niveluri (API si import web);
|
||||
cont pro → trece.
|
||||
|
||||
### US-004: Enforcement DUR — gate API doar pe Pro/Premium
|
||||
**Ca** owner **vreau** ca importul prin API sa fie disponibil doar pe Pro+ **pentru ca** landing-ul
|
||||
spune ca API-ul e o capabilitate Pro.
|
||||
|
||||
- **Depinde de**: US-001
|
||||
- **Fisiere**: `app/auth.py` (sau dependinta de ruta), `app/api/v1/router.py`,
|
||||
`app/api/v1/import_router.py`, `app/errors.py` (cod `PLAN_FARA_API`), `tests/test_api_scope.py`
|
||||
(~5 fisiere)
|
||||
- **Test intai (RED)**: `test_free_fara_api_403`, `test_standard_fara_api_403`, `test_pro_api_ok`,
|
||||
`test_trial_pro_api_ok`, `test_dry_run_valideaza_ramane_permis`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Rutele de **import/ingestie prin API** (`POST /v1/prezentari`, `POST /v1/import`, etc.)
|
||||
cer `effective_tier.api_access == True` (pro/premium sau trial Pro). Altfel 403 cu eroare
|
||||
3 niveluri (`PLAN_FARA_API`: "Importul prin API e disponibil pe planul Pro", fix).
|
||||
- [ ] **Canalul web ramane neafectat** — operatorii pe plan gratuit pot folosi import xlsx/csv prin
|
||||
dashboard (asa promite landing-ul: Gratuit are import manual, NU API). Doar suprafata API e gated.
|
||||
- [ ] `GET /v1/nomenclator` ramane public (coduri RAR, fara PII) — invariant CLAUDE.md.
|
||||
- [ ] `POST /v1/prezentari/valideaza` (dry-run) — decizie: ramane permis pe orice plan (read-only,
|
||||
ajuta integrarea inainte de upgrade) SAU gated ca restul API. Implicit: PERMIS (read-only,
|
||||
fara enqueue). De confirmat.
|
||||
- [ ] In dev (`AUTOPASS_REQUIRE_API_KEY=false`), contul id=1 are acces API (tier coerent), ca testele
|
||||
API existente sa nu pice.
|
||||
- **Verificare E2E**: cheie API pe cont free → 403 onest pe import; cheie pe cont pro/trial → 200.
|
||||
|
||||
### US-005: Downgrade automat la expirarea trial-ului
|
||||
**Ca** owner **vreau** ca la expirarea celor 30 de zile contul sa treaca automat pe Gratuit **pentru ca**
|
||||
landing-ul spune "apoi trece automat pe Gratuit, fara plata".
|
||||
|
||||
- **Depinde de**: US-001, US-003, US-004
|
||||
- **Fisiere**: `app/plans.py` (`effective_tier` deja trateaza expirarea — lazy), optional
|
||||
`app/worker/__main__.py` SAU un job de intretinere (eager), `tests/test_plans.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `test_trial_expirat_efective_free`, `test_trial_expirat_aplica_limita_60`,
|
||||
`test_trial_expirat_pierde_api`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] **Lazy-first**: `effective_tier` returneaza `tier` de baza (`free`) imediat ce
|
||||
`trial_until <= now` — fara job necesar pentru corectitudine (enforcement-ul US-003/004 se
|
||||
bazeaza pe `effective_tier`, deci downgrade-ul e automat la prima cerere dupa expirare).
|
||||
- [ ] Optional (eager, non-blocant): un pas in purjarea orara a worker-ului (T16 existent) poate
|
||||
normaliza `trial_until` expirat → NULL pentru igiena (NU obligatoriu pentru corectitudine).
|
||||
- [ ] Un cont cu `tier='standard'/'pro'/'premium'` setat de admin NU e downgradat de expirarea
|
||||
trial-ului (trial-ul e un BONUS peste `free`; un plan platit alocat persista).
|
||||
- [ ] Mesajele de limita/API dupa expirare sunt cele 3-niveluri din US-003/004.
|
||||
- **Verificare E2E**: setez `trial_until` in trecut → contul aplica limita 60 + pierde API, fara restart.
|
||||
|
||||
### US-006: UI dashboard — plan curent + zile ramase din trial + consum lunar
|
||||
**Ca** operator **vreau** sa vad pe ce plan sunt, cat mi-a mai ramas din trial si cat am consumat
|
||||
luna asta **pentru ca** vreau sa stiu cand ma apropii de limita.
|
||||
|
||||
- **Depinde de**: US-001, US-002
|
||||
- **Fisiere**: `app/web/routes.py` (context), `app/web/templates/_status.html` SAU `_cont.html`
|
||||
(afisaj plan), `tests/test_web_status.py` / `tests/test_dashboard.py` (~4 fisiere)
|
||||
- **Test intai (RED)**: `test_afisaj_plan_si_zile_trial`, `test_afisaj_consum_lunar`,
|
||||
`test_avertizare_aproape_de_limita`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Dashboard-ul afiseaza discret planul curent (ex. "Plan: Pro · trial 18 zile ramase" sau
|
||||
"Plan: Gratuit · 47/60 luna asta"). In trial → eticheta "trial" + zile ramase; pe Gratuit →
|
||||
consum `N/60`.
|
||||
- [ ] **Plasare (aliniat cu PRD 5.16)**: planul apare ca **badge in titlul din antet**
|
||||
(`Gratuit`/`Standard`/`Pro`/`Premium`) SI ca linie in **meniul burger** ("Plan: <tier> [· trial
|
||||
N zile]"), nu doar intr-un card pe Acasa. Vezi mockup-urile 5.16
|
||||
(`docs/mockups/prd-5.16-dashboard.html` / `...-mobil.html`). 5.16 furnizeaza locul de afisare
|
||||
(antet + meniu); 5.17 furnizeaza datele (tier, trial, consum).
|
||||
- [ ] Avertizare vizuala cand consumul Gratuit se apropie de limita (ex. ≥80% → ton warn), fara a
|
||||
ingropa stripul de sanatate (zero-silent-failures pastrat).
|
||||
- [ ] Scoped pe cont; design conform 5.15/5.16 (tokeni, fonturi system, fara hex hardcodat).
|
||||
- [ ] Pagina "Cont" arata planul + (daca exista) o explicatie "cum trec pe alt plan" (contact, ca
|
||||
nu exista plata self-service inca).
|
||||
- **Verificare E2E**: cont trial → "trial N zile"; cont free aproape de 60 → avertizare; cont pro →
|
||||
fara contor de limita.
|
||||
|
||||
### US-007: Aliniere landing — limita 60 + trial pe Pro (nu Premium)
|
||||
**Ca** vizitator **vreau** ca landing-ul sa spuna adevarul **pentru ca** azi promite "100/luna" si
|
||||
"Premium gratuit 30 zile", dar realitatea va fi 60/luna si trial pe Pro.
|
||||
|
||||
- **Depinde de**: — (copy-only; aliniaza cu modelul din US-001)
|
||||
- **Fisiere**: `app/web/templates/landing.html`, `tests/test_web_*` (~2 fisiere)
|
||||
- **Test intai (RED)**: `test_landing_limita_60`, `test_landing_trial_pro_nu_premium`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Toate aparitiile "100 de prestatii/luna" / "100/luna" / `meta description`
|
||||
(`landing.html:7,65,266` + oriunde apar) → **60**. Inclusiv cardul Gratuit din sectiunea PRICING.
|
||||
- [ ] Textul "Fiecare cont incepe cu **Premium gratuit 30 de zile**" (`landing.html:256`) →
|
||||
"**Pro gratuit 30 de zile**" (planul corect). Restul frazei ("Apoi trece automat pe Gratuit…")
|
||||
ramane.
|
||||
- [ ] Coerenta: orice alt loc care implica trial/limita reflecta 60 + Pro.
|
||||
- [ ] Fara alte schimbari de pret/continut (39/59 lei raman).
|
||||
- **Verificare E2E**: landing in browser — "60 prestatii/luna" peste tot, "Pro gratuit 30 de zile".
|
||||
|
||||
### US-008: Admin — alocare manuala de plan (fara plata self-service)
|
||||
**Ca** admin **vreau** sa pot seta planul unui cont **pentru ca** nu exista inca facturare automata,
|
||||
dar trebuie sa pot acorda Standard/Pro/Premium.
|
||||
|
||||
- **Depinde de**: US-001
|
||||
- **Fisiere**: `tools/account.py` (CLI `set-tier`), optional `app/web/routes.py` (`/admin` actiune),
|
||||
`tests/test_accounts.py` / `tests/test_web_admin*.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `test_cli_set_tier`, `test_admin_set_tier_scoped`, `test_tier_invalid_respins`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] CLI `python3 -m tools.account set-tier --account N --tier pro [--trial-days 30|--no-trial]`
|
||||
seteaza `tier`/`trial_until`. Tier invalid → eroare clara.
|
||||
- [ ] Optional (la executie): actiune in panoul `/admin` pentru a seta planul unui cont (scoped,
|
||||
CSRF, ca bulk-ul de status din 5.5). Daca nu intra in 5.17, CLI e suficient (admin-only).
|
||||
- [ ] Alocarea unui plan platit de catre admin NU e suprascrisa de expirarea trial-ului (US-005).
|
||||
- [ ] Audit: schimbarea de plan se logheaza in `app_events` (reuse jurnalul din 5.6), fara PII nou.
|
||||
- **Verificare E2E**: `set-tier --account 2 --tier pro` → contul 2 are API + volum nelimitat.
|
||||
|
||||
### US-009: Teste de regresie + E2E plan/trial/enforcement
|
||||
**Ca** dezvoltator **vreau** acoperire completa **pentru ca** enforcement-ul atinge ambele canale de
|
||||
ingestie si nu vreau sa blochez gresit conturi legitime.
|
||||
|
||||
- **Depinde de**: US-003, US-004, US-005, US-006, US-007
|
||||
- **Fisiere**: `tests/test_plans.py`, `tests/test_api_scope.py`, `tests/test_web_*` (~3 fisiere)
|
||||
- **Test intai (RED)**: matricea plan × capabilitate (volum, API) × canal (API, web) × trial activ/expirat.
|
||||
- **Acceptance criteria**:
|
||||
- [ ] `python3 -m pytest -q -m "not live"` verde; regresia de aur (`POST /v1/prezentari` → queued
|
||||
pe un cont cu drept) ramane verde.
|
||||
- [ ] Matrice testata: free(volum-blocat/API-blocat), standard(volum-ok/API-blocat),
|
||||
pro(ok/ok), trial-pro(ok/ok), trial-expirat(=free).
|
||||
- [ ] Contoarele lunare resetate la luna noua (test la granita timp local RO).
|
||||
- [ ] Dev (id=1) nu e blocat de enforcement (dogfooding).
|
||||
- **Verificare E2E**: rulare completa documentata in Raportul VERIFY.
|
||||
|
||||
## 4. Cerinte functionale (rezumat)
|
||||
1. [REQ-001] `accounts.tier` ∈ {free,standard,pro,premium} + `trial_until`; migrare aditiva defensiva.
|
||||
2. [REQ-002] `app/plans.py` = sursa unica: limite (free=60/luna) + capabilitati (API doar Pro+).
|
||||
3. [REQ-003] Cont nou → trial Pro 30 zile automat; `effective_tier` randeaza Pro in trial, free dupa.
|
||||
4. [REQ-004] Enforcement DUR: free peste 60/luna respins (API + import web) cu eroare 3 niveluri.
|
||||
5. [REQ-005] Enforcement DUR: import API gated pe Pro+ (403 onest); canalul web ramane liber.
|
||||
6. [REQ-006] Downgrade automat la expirare trial (lazy via `effective_tier`).
|
||||
7. [REQ-007] Dashboard arata plan + zile trial + consum lunar; landing aliniat (60, Pro).
|
||||
8. [REQ-008] Admin aloca planuri manual (CLI `set-tier`), audit in `app_events`.
|
||||
|
||||
## 5. Non-Goals (anti scope-creep)
|
||||
- **Fara integrare de plata / facturare / abonamente** (Stripe etc.) — alocarea platita = manuala (admin).
|
||||
- Fara self-service upgrade din UI (doar afisare plan + "contacteaza-ne"); plata vine intr-un PRD viitor.
|
||||
- Fara modificari pe backend-ul de trimitere (worker, masina de stari, idempotenta `build_key`,
|
||||
reconciliere, contract RAR). Enforcement-ul se face la ingestie/enqueue, INAINTE de coada.
|
||||
- Fara schimbarea capabilitatilor de produs in sine (sugestii/mapare exista deja pe toate planurile in
|
||||
cod; diferentierea 5.17 e pe VOLUM + ACCES API, exact ce promite landing-ul ca diferentiator hard).
|
||||
- Fara modificari de design (tipografia/temele sunt 5.16/5.15); doar reuse-ul stilurilor existente.
|
||||
|
||||
## 6. Consideratii tehnice
|
||||
- **Stack**: SQLite (migrare aditiva defensiva ca 5.5/5.12), FastAPI, Jinja2/HTMX.
|
||||
- **Patterns de urmat**: sursa unica (`app/plans.py` ca `app/errors.py`); eroare 3 niveluri (5.4);
|
||||
scope pe cont (5.15/US-011); timp local RO la bucketare (5.15/E7); audit `app_events` (5.6).
|
||||
- **Riscuri**:
|
||||
- **Blocare gresita a unui cont legitim** (enforcement prea agresiv) — risc de business. Mitigare:
|
||||
dev id=1 exceptat; teste matrice; mesaje 3 niveluri cu cale de iesire; respingere INAINTE de enqueue
|
||||
(nu pierde date).
|
||||
- **Definitia "prestatie consumata"** (acceptate-in-coada vs sent) schimba cand musca limita.
|
||||
Mitigare: o decidem explicit (US-002 AC) + test; documentam.
|
||||
- **Granita de luna / fus orar** — off-by-a-day la reset. Mitigare: timp local RO + test la granita
|
||||
(lectia E7 din 5.15).
|
||||
- **Idempotenta vs cota** — un retry idempotent nu trebuie sa consume cota de doua ori. Mitigare:
|
||||
enforce inainte de `build_key`; testul de retry.
|
||||
- **Conturi legacy fara tier** — migrare le pune `free`; un cont real activ ar putea fi limitat brusc
|
||||
la 60. Mitigare: decizie de migrare (legacy activ → ce plan?) confirmata cu user inainte de deploy.
|
||||
|
||||
## 7. Consideratii UI/UX
|
||||
- Afisaj plan discret, conform 5.16 (fonturi system, tokeni `--fs-*`, fara hex).
|
||||
- Stari: trial activ (zile ramase) / free (consum N/60, warn la ≥80%) / platit (fara contor limita).
|
||||
- Mesaje de respingere oneste, actionabile (cum trec pe alt plan), nu doar "403".
|
||||
|
||||
## 8. Open Questions
|
||||
- [ ] "Prestatie consumata" = acceptate-in-coada (queued+sending+sent) sau doar `sent`? (implicit: acceptate)
|
||||
- [ ] Lot care depaseste partial limita → respingere totala sau enqueue partial? (implicit: respingere totala clara)
|
||||
- [ ] `POST /v1/prezentari/valideaza` (dry-run) — gated pe Pro sau permis tuturor? (implicit: permis)
|
||||
- [x] ~~Migrare conturi legacy active: raman `free` sau primesc un trial/plan?~~ **REZOLVAT (user, 2026-06-28): NU exista conturi legacy (produs in TESTE, pre-productie) -> intrebare moot; enforcement DUR direct de la deploy.**
|
||||
- [ ] Standard (39 lei) si Premium difera de Pro doar prin API + suport in landing — pastram exact maparea
|
||||
de capabilitati din landing in `plans.py`? (implicit: da)
|
||||
|
||||
## 9. Valuri de executie
|
||||
|
||||
```
|
||||
Val 1: [US-001] schema tier+trial + app/plans.py (autor unic schema/accounts)
|
||||
Val 2: [US-002] numarator consum lunar (dupa model) ||
|
||||
[US-007] landing copy 60 + Pro (independent, copy-only)
|
||||
Val 3: [US-003] [US-004] [US-005] enforcement volum + API + downgrade (consuma plans.py)
|
||||
Val 4: [US-006] [US-008] UI dashboard plan/consum || admin set-tier
|
||||
Val 5: [US-009] regresie + E2E matrice (dupa toate)
|
||||
```
|
||||
|
||||
> Secventiere fata de 5.16: independent (5.16 = design/tipografie; 5.17 = model de cont). Pot rula in
|
||||
> paralel; doar US-006 (afisaj plan in `_status.html`) atinge un fisier pe care 5.16/US-003 il modifica
|
||||
> (dot RAR) — serializeaza acel template daca ambele PRD-uri sunt in executie simultan.
|
||||
|
||||
---
|
||||
|
||||
> Acest PRD nu a fost inca trecut prin `/plan-ceo-review` / `/plan-eng-review`. Recomandat inainte de
|
||||
> executie (enforcement de business cu risc de blocare gresita + decizia de migrare a conturilor legacy).
|
||||
|
||||
---
|
||||
|
||||
# REVIZIE /autoplan (2026-06-28)
|
||||
|
||||
> Pipeline complet rulat: CEO -> Design -> Eng -> DX. Mod: **SELECTIVE EXPANSION**.
|
||||
> Sesiune spawned (non-interactiva): fiecare AskUserQuestion intermediar a fost auto-decis cu cele
|
||||
> 6 principii; deciziile "taste" si "user challenges" sunt colectate la poarta finala (Faza 4).
|
||||
> **Codex INDISPONIBIL** (limita de utilizare atinsa pana la 2026-07-18) -> toate vocile duale
|
||||
> ruleaza `[codex-unavailable] / [subagent-only]` cu vocea analitica independenta Claude ca model unic.
|
||||
> Restore point: vezi comentariul HTML din capul fisierului.
|
||||
|
||||
## Faza 0 — Intake
|
||||
|
||||
- **Scop UI detectat: DA** (dashboard, badge antet, meniu burger, `_status.html`/`_cont.html`,
|
||||
avertizare vizuala, mockup-uri 5.16) -> Faza 2 (Design) ruleaza.
|
||||
- **Scop DX detectat: DA** (endpointuri `/v1/*`, 403/erori 3-niveluri, CLI `tools.account set-tier`,
|
||||
cheie API, mesaje pentru integratori) -> Faza 3.5 (DX) ruleaza.
|
||||
- Cod citit: `app/accounts.py`, `app/schema.sql` (accounts/submissions/app_events), `app/errors.py`,
|
||||
`app/auth.py` (`resolve_account_id`), `app/api/v1/router.py` (`create_prezentari`/`valideaza`),
|
||||
`app/api/v1/import_router.py` (`commit_import`), `tools/account.py`, `app/web/templates/landing.html`.
|
||||
|
||||
## Faza 1 — CEO Review (Strategie & Scop) [subagent-only]
|
||||
|
||||
### 0B. Ce exista deja (leverage map)
|
||||
| Sub-problema 5.17 | Cod existent reutilizabil | Reuse? |
|
||||
|---|---|---|
|
||||
| Sursa unica de adevar (definitii) | `app/errors.py` (pattern CATALOG + `eroare()`) | DA — `plans.py` copiaza pattern-ul |
|
||||
| Eroare 3 niveluri | `app/errors.py::eroare()` (problema/cauza/fix) | DA — adauga `PLAN_LIMITA_LUNARA`, `PLAN_FARA_API` in CATALOG |
|
||||
| Migrare aditiva defensiva | `_migrate` in `db.py` (ALTER ca `email`/`status` 5.5/5.12) | DA |
|
||||
| Scope pe cont la ingestie | `auth.py::resolve_account_id` (Depends) | DA — gate API se ataseaza aici/ruta |
|
||||
| Lifecycle cont + protectie id=1 | `accounts.py` (`set_status`, `_PROTECTED_ACCOUNT_ID`) | DA — `set_tier` urmeaza acelasi tipar |
|
||||
| Audit fara PII | `observ.py::log_event` -> `app_events` (5.6) | DA — log schimbare plan |
|
||||
| CLI admin | `tools/account.py` (argparse) | DA — subcomanda `set-tier` |
|
||||
| Consum lunar | `submissions.created_at` + `idx_submissions_account_status` | DA — fara coloana noua |
|
||||
|
||||
### 0C. Dream state
|
||||
```
|
||||
CURENT 5.17 IDEAL 12 LUNI
|
||||
landing promite 4 planuri, -> model de cont real (tier+trial), -> facturare self-service
|
||||
app nu stie de tipuri; enforcement volum+API, (Stripe), upgrade din UI,
|
||||
trial inexistent; downgrade lazy la expirare, dunning, conversie masurata,
|
||||
limita 100 doar pe hartie admin manual aloca plan platit re-trial/nurture automat
|
||||
```
|
||||
Delta: 5.17 aliniaza app-ul cu promisiunea landing-ului, DAR ramane fara calea de conversie
|
||||
(plata self-service) — enforcement-ul musca inainte sa existe un buton de upgrade.
|
||||
|
||||
### 0C-bis. Alternative de implementare
|
||||
```
|
||||
APROACH A: Enforcement DUR (planul actual)
|
||||
Rezumat: respinge la enqueue free>60 + 403 API non-Pro; downgrade lazy.
|
||||
Efort: M (human ~2-3z / CC ~45min) Risc: Mediu-Inalt (blocare gresita fara cale de upgrade)
|
||||
Pro: aliniere completa cu landing; diferentiator hard real.
|
||||
Contra: friction fara conversie self-service; risc fals-block legacy.
|
||||
Reuse: errors.py, auth.py, app_events.
|
||||
|
||||
APROACH B: Soft-first (warn + overgrace + flag admin) [recomandat de revizie]
|
||||
Rezumat: la depasire limita -> avertizare clara + enqueue permis cu marcaj, alerta admin;
|
||||
API gate ramane DUR (capability, nu volum). Hard-block volum activabil ulterior prin flag.
|
||||
Efort: M (human ~2-3z / CC ~45min) Risc: Scazut.
|
||||
Pro: zero fals-block; conversie prin contact, nu prin churn; deploy mai sigur.
|
||||
Contra: nu "forteaza" upgrade; cota e mai degraba un semnal decat un zid.
|
||||
Reuse: identic cu A.
|
||||
|
||||
APROACH C: Model + copy now, enforcement sub feature flag (deferat)
|
||||
Rezumat: adauga tier/trial + plans.py + fix landing; enforcement scris dar OFF (flag),
|
||||
pornit dupa migrare legacy confirmata.
|
||||
Efort: S-M Risc: Foarte scazut.
|
||||
Pro: deploy incremental, decuplaza copy-fix (banal) de enforcement (riscant).
|
||||
Contra: promisiunea landing nu e inca "reala" la deploy.
|
||||
```
|
||||
**RECOMANDARE revizie:** combina **C (feature flag de enforcement) + B (soft-first pe VOLUM)**,
|
||||
pastrand **A pe gate-ul API** (capability, risc mic). Principii P1 (completeness pe model) + P6
|
||||
(bias to action: deploy incremental). Vezi TASTE DECISION T-CEO-1 si T-CEO-2 la poarta.
|
||||
|
||||
### 0E. Interogare temporala
|
||||
- HOUR 1 (foundations): valorile exacte ale planurilor (sursa unica `plans.py`); valoarea `60` ca
|
||||
CONSTANTA unica; politica legacy (free fara trial vs trial calculat din `created_at`).
|
||||
- HOUR 2-3 (core): definitia "prestatie consumata" (acceptate-in-coada vs sent); bucketare luna
|
||||
timp local RO (lectia E7/5.15); interactiunea enforce-inainte-de-`build_key` (idempotenta).
|
||||
- HOUR 4-5 (integrare): unde se ataseaza gate-ul API (dependinta de ruta vs in `resolve_account_id`);
|
||||
lot care depaseste partial limita (respingere totala vs partial); `valideaza` dry-run gated sau nu.
|
||||
- HOUR 6+ (polish/teste): matrice plan x capabilitate x canal x trial; granita de luna; dev id=1 exceptat.
|
||||
|
||||
### 0F. Mod: SELECTIVE EXPANSION (default pentru iteratie pe sistem existent). Approach: B+C pe volum, A pe API.
|
||||
|
||||
### Voci duale (CEO)
|
||||
**CODEX SAYS (CEO — strategy challenge):** `[codex-unavailable]` — limita de utilizare (pana 2026-07-18).
|
||||
Voce omisa; consensul se calculeaza N/A pe coloana Codex.
|
||||
|
||||
**CLAUDE SUBAGENT (CEO — strategic independence)** (voce analitica independenta, inainte de orice Codex):
|
||||
1. **Problema corecta?** Gap real: landing-ul promite planuri pe care app-ul nu le sustine. DAR
|
||||
enforcement-ul DUR pe volum apare INAINTEA oricarei cai de plata. Reframe: "onestitate landing +
|
||||
diferentiere capability" se poate atinge fara a ZIDI free-ul la 60. (HIGH)
|
||||
2. **Premise asumate:** (a) "promisiunile trebuie impuse DUR acum" — asumata; un fix de copy + gate API
|
||||
ar inchide 80% din gap cu 20% din risc. (b) "60 in loc de 100" — decizie user, dar fara rationament;
|
||||
scade atractivitatea free-ului exact cand nu exista upgrade self-service. (MEDIUM)
|
||||
3. **Regret la 6 luni:** un cont free real face 80/luna, e migrat la free si blocat brusc la 60 ->
|
||||
churn in loc de conversie (nu exista buton de upgrade, doar "contacteaza-ne"). (HIGH, deploy-blocker
|
||||
pe migrarea legacy.)
|
||||
4. **Alternative neexplorate:** soft-enforcement (warn+overgrace) vs hard-block; planul sare direct la hard.
|
||||
5. **Risc competitiv:** nisa B2B reglementata (RAR), switching cost real -> risc competitiv scazut;
|
||||
riscul dominant e INTERN (friction fara conversie).
|
||||
|
||||
```
|
||||
CEO DUAL VOICES — CONSENSUS TABLE:
|
||||
═══════════════════════════════════════════════════════════════
|
||||
Dimensiune Claude Codex Consensus
|
||||
───────────────────────────────────── ─────── ─────── ─────────
|
||||
1. Premise valide? Partial N/A N/A (Codex indisp.)
|
||||
2. Problema corecta? Da* N/A N/A
|
||||
3. Calibrare scop corecta? Nu** N/A N/A
|
||||
4. Alternative explorate suficient? Nu N/A N/A
|
||||
5. Riscuri piata acoperite? Da N/A N/A
|
||||
6. Traiectorie 6 luni sanatoasa? Partial N/A N/A
|
||||
═══════════════════════════════════════════════════════════════
|
||||
* problema reala, dar solutia (hard enforce) e mai agresiva decat o cere problema.
|
||||
** scop corect ca model; enforcement-ul DUR pe volum e calibrat prea agresiv pentru un produs fara plata.
|
||||
Single-model: niciun consens incrucisat; constatarile critice ale vocii Claude sunt semnalate oricum.
|
||||
```
|
||||
|
||||
### Sectiunile 1-11 (CEO)
|
||||
|
||||
**S1 Arhitectura.** Componenta noua `plans.py` = modul PUR (ca `errors.py`), fara import DB/HTTP, dict
|
||||
`PLANS` + `effective_tier(account_row, now)` + `monthly_usage(conn, account_id, now)`. Cuplare noua:
|
||||
rutele de ingestie (`router.py`, `import_router.py`, `routes.py` commit) depind de `plans.py` + citesc
|
||||
`accounts.tier/trial_until` -> cuplare justificata (un singur punct de adevar). Diagrama: vezi Faza 3 (Eng).
|
||||
Constatare CEO-S1-1 (MEDIUM): `effective_tier` are nevoie de `now` injectabil (nu `datetime.now()` intern)
|
||||
ca testele de granita trial/luna sa fie deterministe. Auto-decis (P5 explicit): semnatura cu `now` parametru.
|
||||
|
||||
**S2 Error & Rescue (registry mai jos).** Coduri noi: `PLAN_LIMITA_LUNARA`, `PLAN_FARA_API`. Ambele
|
||||
sunt erori de business (nu exceptii) -> 3 niveluri din `errors.py`, returnate ca raspuns structurat
|
||||
(nu 500). Fara catch-all. Constatare CEO-S2-1 (LOW): trial expirat NU e o eroare — e o stare; nu necesita
|
||||
cod de eroare, doar `effective_tier` care vede `free`.
|
||||
|
||||
**S3 Securitate (detaliu in Eng S3).** Suprafata: gate API (autorizare pe capability) + enforce volum.
|
||||
DOR (direct object reference) la `set-tier` admin: trebuie scoped + protejat id=1 (ca `set_status`).
|
||||
Risc privilege: un cont free NU trebuie sa-si poata seta singur tier (doar admin CLI / panou admin CSRF).
|
||||
Constatare CEO-S3-1 (HIGH): enforce pe volum/API trebuie sa ruleze DUPA `resolve_account_id` (cont
|
||||
autenticat), niciodata pe baza unui camp din body. Auto-decis (P1): gate ca dependinta server-side.
|
||||
|
||||
**S4 Data flow & edge cases.** Granita de luna (timp local RO), idempotenta vs cota (retry nu consuma
|
||||
de 2x), lot care depaseste partial. Vezi Failure Modes Registry. Edge: 2 cereri concurente la 59/60 ->
|
||||
race pe cota (ambele trec checkul, ajung la 61). Constatare CEO-S4-1 (MEDIUM): cota nu e tranzactionala
|
||||
cu enqueue -> mic overshoot posibil sub concurenta. Auto-decis (P3 pragmatic): accepta overshoot mic
|
||||
(±lot) documentat; un lock per-cont ar fi over-engineering pentru un cap soft. (Daca se alege hard-block,
|
||||
re-evalueaza.)
|
||||
|
||||
**S5 Code quality.** `plans.py` sursa unica evita DRY-violation intre backend si UI. Risc: valoarea `60`
|
||||
sa fie hardcodata in 3 locuri (router, import, web). Auto-decis (P4 DRY): O singura definitie in `PLANS`,
|
||||
consumata peste tot; templating UI primeste `monthly_limit` din context, nu literal.
|
||||
|
||||
**S6 Teste (diagrama in Eng S3).** Matrice plan x capabilitate x canal x trial. Gap-uri critice: granita
|
||||
luna timp local RO; retry idempotent; dev id=1 ne-blocat. Toate cerute in US-009.
|
||||
|
||||
**S7 Performanta.** `monthly_usage` = un COUNT cu `WHERE account_id=? AND status IN (...) AND created_at>=...`.
|
||||
Exista `idx_submissions_account_status(account_id,status)` dar NU acopera `created_at`. Constatare CEO-S7-1
|
||||
(MEDIUM): la volume mari un COUNT pe luna per-cerere e O(randuri luna); acceptabil la scara curenta, dar
|
||||
indexul nu acopera intervalul de timp. Auto-decis (P3): acceptabil acum (SQLite, volume mici); TODO index
|
||||
`(account_id, created_at)` daca apar conturi cu mii/luna. -> TODOS.
|
||||
|
||||
**S8 Observabilitate.** Fiecare respingere pe plan (volum/API) trebuie sa emita `app_events`
|
||||
(cod + cont + count), nu doar sa intoarca 4xx. Altfel "de ce a fost blocat clientul X?" e invizibil.
|
||||
Auto-decis (P_prime zero-silent-failures): log_event pe fiecare respingere de plan. (Adaugat ca AC.)
|
||||
|
||||
**S9 Deploy.** Migrare aditiva defensiva (idempotenta). **REZOLVAT (decizie user 2026-06-28):**
|
||||
enforcement DUR direct de la deploy — fara conturi legacy, produs in TESTE (pre-productie), deci riscul
|
||||
de fals-block e moot. Feature-flag `AUTOPASS_ENFORCE_PLANS` ramane **OPTIONAL** (nice-to-have de operare,
|
||||
kill-switch), NU blocant pentru deploy. Vezi T-CEO-1 (rezolvat).
|
||||
|
||||
**S10 Traiectorie.** Reversibilitate 4/5 (model aditiv; enforcement sub flag = usor de oprit). Path
|
||||
dependency: fara billing, `set-tier` manual devine gatuire daca adoptia creste -> Phase 2 = plata
|
||||
self-service. Datorie: cuplarea enforcement de ingestie e curata; datoria reala e "lipsa caii de upgrade".
|
||||
|
||||
**S11 Design & UX (deep in Faza 2).** Plasare badge plan in antet + meniu burger (aliniat 5.16),
|
||||
avertizare la >=80%, mesaje oneste cu cale de iesire. Recomand /plan-design-review (rulat ca Faza 2).
|
||||
|
||||
### Iesiri obligatorii CEO
|
||||
|
||||
**NOT in scope (deferat, cu rationament):**
|
||||
- Integrare plata/facturare (Stripe) — non-goal explicit; Phase 2.
|
||||
- Upgrade self-service din UI — depinde de billing; doar afisaj + "contacteaza-ne".
|
||||
- Index `(account_id, created_at)` — deferat pana apar conturi de volum mare (TODO P3).
|
||||
- Job eager de normalizare `trial_until` expirat -> NULL — optional, igiena; lazy acopera corectitudinea.
|
||||
- Diferentiere capability de produs (sugestii/mapare) pe planuri — non-goal; diferentierea e volum+API.
|
||||
|
||||
**What already exists:** vezi tabelul 0B (errors.py, auth.py, accounts.py, observ/app_events, db._migrate,
|
||||
submissions.created_at + index, tools/account.py — toate reutilizate; 5.17 nu reconstruieste nimic).
|
||||
|
||||
**Dream state delta:** 5.17 face promisiunea landing-ului REALA in app, dar lasa golul "conversie
|
||||
self-service"; urmatorul pas logic e billing (Phase 2). Enforcement-ul fara upgrade self-service e
|
||||
delta-ul de risc.
|
||||
|
||||
### Error & Rescue Registry (S2)
|
||||
```
|
||||
CODEPATH | CE POATE ESUA | COD / EXCEPTIE
|
||||
---------------------------------|--------------------------------|------------------------
|
||||
create_prezentari (enqueue) | free peste 60/luna | PLAN_LIMITA_LUNARA (business)
|
||||
commit_import (web+API) | free peste 60/luna | PLAN_LIMITA_LUNARA (business)
|
||||
import API / POST /v1/prezentari | cont fara api_access (non-Pro) | PLAN_FARA_API (403, business)
|
||||
effective_tier(account, now) | trial_until malformat/NULL | trateaza ca free (fallback)
|
||||
monthly_usage(conn, acct, now) | created_at NULL/malformat | exclus din count (defensiv)
|
||||
set-tier (CLI/admin) | tier invalid | ValueError -> mesaj clar
|
||||
set-tier pe id=1 | mutare cont sistem | protejat (ca set_status)
|
||||
|
||||
COD / STARE | RESCUED? | ACTIUNE | USER VEDE
|
||||
------------------------|----------|----------------------------------|---------------------------
|
||||
PLAN_LIMITA_LUNARA | Y | respinge inainte de build_key | "Ai atins limita Gratuit (60/luna)" + fix
|
||||
PLAN_FARA_API | Y | 403 inainte de procesare | "Importul API e pe Pro" + fix
|
||||
trial_until malformat | Y | fallback free, log WARNING | comportament free (fara crash)
|
||||
created_at malformat | Y | exclus din count, log WARNING | nimic (transparent)
|
||||
tier invalid (set-tier) | Y | ValueError, exit!=0 | "tier invalid: X"
|
||||
```
|
||||
|
||||
### Failure Modes Registry
|
||||
```
|
||||
CODEPATH | FAILURE MODE | RESCUED? | TEST? | USER VEDE | LOGGED?
|
||||
--------------------------|--------------------------|----------|-------|------------------|--------
|
||||
enforce volum (enqueue) | free peste 60 | Y | Y | eroare 3 niveluri| Y (app_events)
|
||||
enforce volum | race concurent la 59/60 | Partial | Y(*) | overshoot mic | Y
|
||||
gate API | non-Pro pe /v1 import | Y | Y | 403 onest | Y
|
||||
downgrade lazy | trial expirat | Y | Y | aplica free | N (stare, nu eveniment)
|
||||
migrare legacy | cont activ -> free brusc | N/A(MOOT)| n/a | n/a | n/a
|
||||
bucketare luna | granita timp local RO | Y | Y | reset corect | n/a
|
||||
idempotenta vs cota | retry consuma cota 2x | Y | Y | nimic | n/a
|
||||
```
|
||||
**~~CRITICAL GAP~~ REZOLVAT (MOOT, 2026-06-28):** decizia userului — NU exista conturi legacy, produsul
|
||||
e in TESTE (pre-productie). Migrarea unui cont activ -> free brusc nu se poate produce (nu exista conturi
|
||||
reale de migrat). Gap inchis ca N/A. Enforcement DUR de la deploy, fara mitigare necesara.
|
||||
|
||||
### Completion Summary (CEO)
|
||||
```
|
||||
+====================================================================+
|
||||
| MEGA PLAN REVIEW — COMPLETION SUMMARY (CEO) |
|
||||
+====================================================================+
|
||||
| Mode | SELECTIVE EXPANSION |
|
||||
| Approach ales | B+C pe volum, A pe gate API |
|
||||
| S1 Arhitectura | 1 (now injectabil) |
|
||||
| S2 Errors | 2 coduri noi, 0 GAP-uri rescue |
|
||||
| S3 Securitate | 1 HIGH (gate server-side), DOR set-tier |
|
||||
| S4 Data/UX | 1 race cota (overshoot mic acceptat) |
|
||||
| S5 Quality | 1 (DRY pe valoarea 60) |
|
||||
| S6 Teste | matrice ceruta, 3 gap-uri acoperite US-009 |
|
||||
| S7 Perf | 1 (index timp) -> TODO |
|
||||
| S8 Observ | 1 (log pe respingere plan) -> AC nou |
|
||||
| S9 Deploy | enforcement DUR direct (user); flag optional |
|
||||
| S10 Future | Reversibilitate 4/5; datorie = lipsa billing|
|
||||
| S11 Design | -> Faza 2 |
|
||||
| NOT in scope | scris (5 items) |
|
||||
| Failure modes | 7 total, 0 CRITICAL GAP (legacy REZOLVAT moot)|
|
||||
| Outside voice | codex indisponibil (subagent-only) |
|
||||
| Unresolved decisions | 0 (toate inchise 2026-06-28: challenge + 3 taste)|
|
||||
+====================================================================+
|
||||
```
|
||||
|
||||
**Phase 1 complete.** Codex: indisponibil. Claude subagent: 9 constatari (2 HIGH, 5 MEDIUM, 2 LOW) +
|
||||
1 USER CHALLENGE + 2 TASTE. Consens: N/A (single-model). Trec la Faza 2.
|
||||
|
||||
## Faza 2 — Design Review [subagent-only]
|
||||
|
||||
> Scop UI confirmat. 5.17 aduce DATELE (tier/trial/consum); 5.16 aduce LOCUL (antet + meniu burger).
|
||||
> Aceasta revizie e la nivel de plan (intentionalitate de design), nu audit de pixeli.
|
||||
> Completitudine design initiala: **6/10** (plasare numita, dar stari incomplete + copy nespecificat).
|
||||
|
||||
**CODEX SAYS (design — UX challenge):** `[codex-unavailable]`.
|
||||
|
||||
**CLAUDE SUBAGENT (design — independent review):**
|
||||
1. **Ierarhie informatie:** badge plan in antet e corect (status, nu actiune); consumul `N/60` apartine
|
||||
contextului secundar (meniu/Cont), NU trebuie sa concureze cu stripul de sanatate. OK.
|
||||
2. **Stari lipsa:** planul numeste "trial activ / free consum / platit fara contor" dar NU specifica:
|
||||
(a) ULTIMA zi de trial ("expira azi" vs "1 zi"), (b) starea "limita ATINSA" (60/60, nu doar >=80%),
|
||||
(c) ce vede operatorul in MOMENTUL respingerii (toast? banner persistent?). GAP (HIGH).
|
||||
3. **Arc emotional:** trial -> "ai Pro 18 zile" (pozitiv) -> ziua 30 trecere tacuta pe free -> prima
|
||||
respingere la 61 = surpriza negativa daca nu a existat avertizare progresiva. Avertizarea >=80% e
|
||||
buna; lipseste un semnal la trecerea trial->free (ziua 0). GAP (MEDIUM).
|
||||
4. **Specificitate vs generic:** "afiseaza discret planul" e generic; mockup-urile 5.16 dau forma, dar
|
||||
copy-ul exact al badge-ului ("Pro · trial 18 zile" / "Gratuit · 47/60") trebuie fixat ca string-uri,
|
||||
nu lasat implementatorului. GAP (MEDIUM).
|
||||
5. **Decizii care vor bantui implementatorul:** prag exact warn (>=80% = 48/60?), pluralizare RO
|
||||
("1 zi" vs "18 zile", "1 zile" e gresit), ce se intampla la 0 zile ramase in trial in aceeasi zi.
|
||||
|
||||
```
|
||||
DESIGN LITMUS SCORECARD (0-10):
|
||||
Dimensiune Claude Codex Consensus
|
||||
────────────────────────────────── ─────── ─────── ─────────
|
||||
1. Ierarhie informatie 8 N/A N/A
|
||||
2. Acoperire stari (load/empty/err) 5 N/A N/A <- gap
|
||||
3. Coerenta user journey 6 N/A N/A
|
||||
4. Specificitate (nu generic) 5 N/A N/A <- gap
|
||||
5. Aliniere design system (5.15/16) 8 N/A N/A
|
||||
6. Intentie responsive 7 N/A N/A
|
||||
7. Accesibilitate (contrast/kbd) 6 N/A N/A
|
||||
────────────────────────────────── ─────── ─────── ─────────
|
||||
Overall design (plan-level) ~6.4/10
|
||||
```
|
||||
|
||||
### Pass-uri 1-7 (constatari + auto-decizii)
|
||||
- **P1 Ierarhie:** badge in antet (status), consum in meniu/Cont. OK, fara modificare.
|
||||
- **P2 Stari (CRITIC):** adauga stari explicite: `trial-activ(N zile)`, `trial-ultima-zi`,
|
||||
`free-sub-prag`, `free-warn(>=80%)`, `free-limita-atinsa(60/60)`, `platit(fara contor)`. Auto-decis
|
||||
(P1 completeness): toate 6 stari intra ca AC in US-006. Matrice stare->afisaj in plan.
|
||||
- **P3 Journey:** adauga un semnal one-time la trecerea trial->free (banner discret "Trial Pro
|
||||
expirat — esti pe Gratuit, 60/luna"). Auto-decis (P1): adaugat ca AC optional in US-006 (non-blocant
|
||||
daca lazy; afisat la prima incarcare dupa expirare). TASTE T-DES-1 (banner one-time vs doar badge).
|
||||
- **P4 Specificitate:** fixeaza string-urile de copy exact (RO, cu pluralizare corecta) in US-006.
|
||||
Auto-decis (P5 explicit): tabel de copy in plan (vezi mai jos).
|
||||
- **P5 Design system:** tokeni `--fs-*`, fonturi system, fara hex hardcodat (5.16). OK; reuse `_status.html`.
|
||||
- **P6 Responsive:** badge in antet + linie in burger acopera desktop+mobil (mockup-uri 5.16). OK.
|
||||
- **P7 Accesibilitate:** tonul "warn" NU doar prin culoare (adauga text/icon); contrast pe badge;
|
||||
badge-ul nu e buton (status) -> fara rol interactiv inselator. Auto-decis (P1): warn = culoare + text.
|
||||
|
||||
**Copy fix (RO, propus, auto-decis P5):**
|
||||
```
|
||||
trial activ: "Plan: Pro · trial {n} {zi|zile} ramase" (1->"zi", 2+->"zile")
|
||||
trial ultima zi: "Plan: Pro · trial expira azi"
|
||||
free sub prag: "Plan: Gratuit · {u}/60 luna asta"
|
||||
free warn (>=80%): "Plan: Gratuit · {u}/60 — aproape de limita"
|
||||
free limita atinsa: "Plan: Gratuit · 60/60 — limita atinsa"
|
||||
platit: "Plan: {Standard|Pro|Premium}"
|
||||
```
|
||||
|
||||
**Required: user flow ASCII (stari + tranzitii)**
|
||||
```
|
||||
[cont nou] --create--> (TRIAL Pro: badge "trial N zile") --N scade zilnic-->
|
||||
(trial ultima zi) --trial_until<=now (lazy)--> (FREE sub prag: "u/60")
|
||||
--u>=48--> (FREE warn ">=80%") --u==60--> (FREE limita atinsa "60/60")
|
||||
|
|
||||
a 61-a cerere -> RESPINS (eroare 3 niveluri / toast)
|
||||
(admin set-tier pro) --------------------------------> (PLATIT: fara contor)
|
||||
```
|
||||
|
||||
**Phase 2 complete.** Codex: indisponibil. Claude subagent: 4 constatari design (1 HIGH stari, 2 MEDIUM,
|
||||
1 accesibilitate) + 1 TASTE (T-DES-1). Overall ~6.4/10 -> tinta dupa AC-uri ~8.5/10. Trec la Faza 3.
|
||||
|
||||
## Faza 3 — Eng Review (Arhitectura & Teste) [subagent-only]
|
||||
|
||||
### Step 0 — Scope challenge (cod citit)
|
||||
- `app/errors.py`: CATALOG + `eroare(cod, field, cauza)` -> pattern de copiat exact pentru coduri noi.
|
||||
- `app/auth.py`: `resolve_account_id` (Depends) intoarce `account_id`; gate-ul API se ataseaza ca a doua
|
||||
dependinta (`require_api_access`) care reuseaza `account_id` -> nu reimplementa auth.
|
||||
- `app/api/v1/router.py`: `create_prezentari` itereaza prestatiile, face `canonicalize_row` -> `build_key`
|
||||
-> enqueue. Gate-ul de VOLUM trebuie INAINTE de bucla de `build_key`/enqueue (idempotenta intacta).
|
||||
- `app/api/v1/import_router.py`: `commit_import` face enqueue per-rand cu ON CONFLICT DO NOTHING; gate
|
||||
volum la inceputul commit-ului (nr randuri `ok` vs cota ramasa).
|
||||
- `app/accounts.py`: `set_status` + `_PROTECTED_ACCOUNT_ID=1` -> `set_tier` urmeaza acelasi tipar (validare
|
||||
tier, protectie id=1, update). `create_account` adauga `tier='free'` + `trial_until=now+30z`.
|
||||
- `tools/account.py`: argparse; adauga subparser `set-tier`.
|
||||
- Complexitate: ramane sub 8 fisiere de logica + `plans.py` nou. Sub pragul de smell. OK.
|
||||
|
||||
**CLAUDE SUBAGENT (eng — independent review):**
|
||||
1. **Arhitectura:** `plans.py` PUR + consum din rute = curat. Singura cuplare noua justificata.
|
||||
2. **Edge:** race pe cota sub concurenta (overshoot ±lot); `now` trebuie injectabil pentru teste de granita.
|
||||
3. **Teste:** matricea e ceruta, dar lipsesc explicit: testul de retry idempotent care NU re-consuma cota,
|
||||
si testul ca `valideaza` dry-run NU consuma cota. (HIGH — sunt invariante usor de stricat.)
|
||||
4. **Securitate:** gate API server-side (nu din body); `set-tier` scoped + protejat id=1.
|
||||
5. **Complexitate ascunsa:** definitia "prestatie consumata" + bucketarea lunii timp local RO sunt sursa
|
||||
reala de bug-uri (off-by-a-day, status care iese din count cand un rand devine `error`).
|
||||
|
||||
```
|
||||
ENG DUAL VOICES — CONSENSUS TABLE:
|
||||
═══════════════════════════════════════════════════════════════
|
||||
Dimensiune Claude Codex Consensus
|
||||
───────────────────────────────────── ─────── ─────── ─────────
|
||||
1. Arhitectura sanatoasa? Da N/A N/A
|
||||
2. Acoperire teste suficienta? Partial N/A N/A
|
||||
3. Riscuri performanta tratate? Partial N/A N/A
|
||||
4. Amenintari securitate acoperite? Da N/A N/A
|
||||
5. Cai de eroare tratate? Da N/A N/A
|
||||
6. Risc deploy gestionabil? Partial N/A N/A (flag + legacy)
|
||||
═══════════════════════════════════════════════════════════════
|
||||
Single-model (codex indisponibil).
|
||||
```
|
||||
|
||||
### Section 1 — Architecture (ASCII)
|
||||
```
|
||||
┌─────────────────────┐
|
||||
│ app/plans.py (NOU) │ modul PUR (ca errors.py)
|
||||
│ PLANS{tier->limite} │ effective_tier(acct,now)
|
||||
│ api_access, limita │ monthly_usage(conn,acct,now)
|
||||
└──────────┬──────────┘
|
||||
┌───────────────┬───────┼───────────────┬──────────────────┐
|
||||
▼ ▼ ▼ ▼ ▼
|
||||
api/v1/router.py import_router web/routes.py auth.py web/templates
|
||||
create_prezentari commit_import commit web require_api_access _status/_cont.html
|
||||
│ gate VOLUM │ gate VOLUM │ gate VOLUM │ gate API (403) badge plan
|
||||
▼ ▼ ▼ ▼
|
||||
errors.eroare(PLAN_LIMITA_LUNARA / PLAN_FARA_API) observ.log_event(app_events)
|
||||
│
|
||||
▼ (daca trece)
|
||||
canonicalize_row -> build_key -> enqueue submissions <-- NESCHIMBAT (worker/idempotenta/reconcile)
|
||||
|
||||
accounts.py: create_account(tier='free', trial_until=now+30z) ; set_tier(acct,tier,trial)
|
||||
db._migrate: ALTER accounts ADD tier / trial_until (aditiv defensiv, idempotent)
|
||||
tools/account.py: subcomanda set-tier
|
||||
config.py: AUTOPASS_ENFORCE_PLANS (flag, vezi T-CEO-1)
|
||||
```
|
||||
Cuplare before/after: inainte rutele depind doar de auth+idempotency+validation; dupa adauga o dependinta
|
||||
catre `plans.py` (pur, fara cicluri). Single point of failure: niciunul nou (modul pur, fara IO).
|
||||
Rollback: revert + flag OFF; migrarea e aditiva (coloanele raman, inofensive).
|
||||
|
||||
### Section 2 — Code quality
|
||||
- DRY: valoarea 60 + maparea capability EXCLUSIV in `PLANS`. Constatare ENG-S2-1: nu duplica `status IN
|
||||
(...)` (definitia consumului) intre `monthly_usage` si teste — exporta o constanta `CONSUMED_STATUSES`.
|
||||
- Naming: `effective_tier`, `monthly_usage`, `api_access`, `monthly_limit` — clare.
|
||||
- Over/under-engineering: NU adauga tabela `plan_usage` (coloana noua) — `submissions.created_at` ajunge
|
||||
(respecta non-goal migrare minima). Lock per-cont pe cota = over-engineering pentru cap soft.
|
||||
|
||||
### Section 3 — Test Review (diagrama completa — NU se sare)
|
||||
```
|
||||
NEW DATA FLOWS:
|
||||
- cerere ingestie -> citeste effective_tier -> compara monthly_usage+nr vs limita -> permite/respinge
|
||||
- cont nou -> create_account seteaza trial_until
|
||||
- trial_until <= now -> effective_tier randeaza free (lazy)
|
||||
NEW CODEPATHS / BRANCHES:
|
||||
- tier in {free,standard,pro,premium}; api_access T/F; monthly_limit None/60
|
||||
- effective_tier: trial activ vs expirat vs plan platit (nu downgrada)
|
||||
- enforce volum: sub limita / la limita / peste / lot care depaseste partial
|
||||
- gate API: free/standard -> 403 ; pro/premium/trial -> ok ; nomenclator public ; valideaza permis
|
||||
- dev id=1: ne-blocat (AUTOPASS_REQUIRE_API_KEY=false)
|
||||
NEW INTEGRATIONS/EXTERNAL: niciuna (totul intern; worker/RAR neatins)
|
||||
NEW ERROR/RESCUE: PLAN_LIMITA_LUNARA, PLAN_FARA_API (+ log_event)
|
||||
|
||||
ITEM | TIP TEST | EXISTA? | HAPPY / FAIL / EDGE
|
||||
--------------------------------------|--------------|---------|---------------------------------
|
||||
migrare tier+trial defensiva | unit (db) | NOU | re-rulare idempotenta; legacy->free
|
||||
PLANS definitii + capability map | unit | NOU | free=60/noAPI; pro=None/API
|
||||
effective_tier trial activ/expirat | unit (now inj)| NOU | viitor->pro; trecut->free; platit persista
|
||||
monthly_usage count | unit | NOU | numara queued+sending+sent; reset luna noua
|
||||
monthly_usage granita timp local RO | unit | NOU | rand la 23:30 UTC ultima zi -> luna RO corecta
|
||||
enforce volum free>60 API | integration | NOU | a 61-a respinsa 3 niveluri
|
||||
enforce volum free>60 import web | integration | NOU | commit respins peste cota
|
||||
enforce volum lot partial | integration | NOU | 50 folosite + lot 20 -> respingere totala (default)
|
||||
retry idempotent NU re-consuma cota | integration | NOU | <-INVARIANT critic
|
||||
valideaza dry-run NU consuma cota | integration | NOU | <-INVARIANT critic
|
||||
gate API free/standard 403 | integration | NOU | 403 onest
|
||||
gate API pro/trial 200 | integration | NOU | trece
|
||||
nomenclator public ramane | integration | reuse | fara cheie -> 200
|
||||
dev id=1 ne-blocat | integration | NOU | dogfooding nu pica
|
||||
set-tier CLI + invalid + id=1 protejat| unit | NOU | tier ok; invalid err; id=1 respins
|
||||
regresie aur (POST -> queued) | integration | reuse | ramane verde
|
||||
```
|
||||
Test 2am-Friday: "un cont Pro NU e blocat niciodata pe volum, indiferent de consum". Test ostil:
|
||||
"trimit 100 cereri concurente la 59/60 pe free" -> verifica overshoot marginit + log. Flakiness: testele
|
||||
de granita luna/trial trebuie sa injecteze `now` (fara `datetime.now()` intern) — altfel flaky.
|
||||
LLM/eval: 5.17 NU atinge prompturi/mapare LLM -> fara eval suites (confirmat: non-goal pe backend trimitere).
|
||||
|
||||
### Section 4 — Performance
|
||||
- `monthly_usage`: COUNT per-cerere; index `(account_id,status)` exista, NU acopera `created_at`.
|
||||
ENG-S4-1 (MEDIUM): la conturi de volum mare scaneaza randurile lunii. Auto-decis (P3): acceptabil acum;
|
||||
TODO index `(account_id, created_at)` (P3) cand apar conturi cu mii/luna.
|
||||
- Fara N+1, fara conexiuni noi, fara job nou (downgrade = lazy).
|
||||
|
||||
### Iesiri obligatorii Eng
|
||||
**NOT in scope (eng):** tabela `plan_usage` dedicata (nu necesara); lock tranzactional pe cota (overshoot
|
||||
mic acceptat); job eager downgrade (lazy ajunge); index timp (TODO).
|
||||
**What already exists (eng):** errors.eroare, auth.resolve_account_id, accounts.set_status pattern,
|
||||
db._migrate, observ.log_event, idempotency.build_key/canonicalize_row, submissions index — toate reutilizate.
|
||||
**Failure modes (eng) cu gap critic:** vezi Failure Modes Registry (CEO) — singurul CRITICAL GAP =
|
||||
migrare legacy active (acoperit de flag + decizie user T-CEO-1).
|
||||
|
||||
### Completion Summary (Eng)
|
||||
```
|
||||
| S1 Arhitectura | curata, 1 cuplare justificata, diagrama produsa |
|
||||
| S2 Quality | 1 (CONSUMED_STATUSES constanta) |
|
||||
| S3 Teste | diagrama produsa; 2 invariante critice (retry, dry-run) |
|
||||
| S4 Perf | 1 (index timp -> TODO P3) |
|
||||
| Artifact teste | scris in ~/.gstack/projects/romfast-rar-autopass/ |
|
||||
| Critical gaps | 1 (legacy) -> flag + decizie user |
|
||||
| Outside voice | codex indisponibil (subagent-only) |
|
||||
```
|
||||
|
||||
**Phase 3 complete.** Codex: indisponibil. Claude subagent: 4 constatari (1 HIGH teste-invariante,
|
||||
3 MEDIUM). Artifact test-plan scris pe disc. Trec la Faza 3.5 (DX).
|
||||
|
||||
## Faza 3.5 — DX Review [subagent-only]
|
||||
|
||||
> Scop DX confirmat: integratorul ROAAUTO/soft propriu foloseste `/v1/*` cu cheie API; adminul foloseste
|
||||
> CLI `tools.account`. Tip produs: **gateway API B2B + CLI admin**. Persona: dezvoltator integrator RO
|
||||
> (consuma `POST /v1/prezentari`) + admin gateway.
|
||||
|
||||
**CODEX SAYS (DX — developer experience challenge):** `[codex-unavailable]`.
|
||||
|
||||
**CLAUDE SUBAGENT (DX — independent review):**
|
||||
1. **Time-to-hello-world:** neschimbat de 5.17 pentru cont cu drept; DAR un integrator pe cont free care
|
||||
incearca `POST /v1/prezentari` va primi acum 403 (PLAN_FARA_API) la primul apel. Daca mesajul nu spune
|
||||
clar "API e pe Pro, dar `valideaza` merge", dezvoltatorul crede ca integrarea e stricata. (HIGH)
|
||||
2. **Mesaje de eroare:** `PLAN_FARA_API` si `PLAN_LIMITA_LUNARA` trebuie problema+cauza+fix (au structura
|
||||
din errors.py). Fix-ul trebuie sa fie actionabil ("Treci pe Pro: contacteaza-ne / set-tier"), nu doar 403.
|
||||
3. **API/CLI naming:** `set-tier --tier pro --trial-days 30|--no-trial` e consistent cu `tools.account`
|
||||
existent (create/activate/deactivate). OK. Sugestie: si `--account` (deja folosit).
|
||||
4. **Docs:** `/v1/nomenclator` ramane public (bun pentru explorare pre-upgrade). `valideaza` permis pe orice
|
||||
plan = excelent DX (integrezi+testezi inainte de a plati). Trebuie documentat explicit ca "poti dezvolta
|
||||
pe free cu valideaza, dar trimiterea reala cere Pro".
|
||||
5. **Upgrade path:** fara self-service -> 403 zice "contacteaza-ne"; un dezvoltator vrea un link/email
|
||||
concret, nu "contact". (MEDIUM)
|
||||
|
||||
```
|
||||
DX DUAL VOICES — CONSENSUS TABLE:
|
||||
Dimensiune Claude Codex Consensus
|
||||
───────────────────────────────────── ─────── ─────── ─────────
|
||||
1. Getting started < 5 min? Da* N/A N/A (*free->403 surprinde)
|
||||
2. Naming API/CLI ghicibil? Da N/A N/A
|
||||
3. Mesaje de eroare actionabile? Partial N/A N/A
|
||||
4. Docs gasibile & complete? Partial N/A N/A
|
||||
5. Upgrade path sigur? Partial N/A N/A (fara self-service)
|
||||
6. Mediu dev fara friction? Da N/A N/A (valideaza permis)
|
||||
```
|
||||
|
||||
### Developer journey map (9 etape)
|
||||
| Etapa | Azi | Cu 5.17 | Friction |
|
||||
|---|---|---|---|
|
||||
| 1 Descoperire | landing | landing aliniat (60, Pro) | — |
|
||||
| 2 Signup | cont + trial Pro | trial Pro 30z automat | — |
|
||||
| 3 Cheie API | CLI apikey | idem | — |
|
||||
| 4 Primul apel | 200 | 200 in trial; 403 pe free dupa trial | mesaj clar necesar |
|
||||
| 5 Dezvoltare | — | `valideaza` permis pe orice plan | excelent |
|
||||
| 6 Trimitere reala | 200 | gated pe Pro+ | upgrade path |
|
||||
| 7 Atingere limita | — | free 60/luna -> respins | mesaj 3 niveluri |
|
||||
| 8 Upgrade | — | contact admin (fara self-service) | link concret |
|
||||
| 9 Operare | dashboard | + badge plan/consum | — |
|
||||
|
||||
### Developer empathy narrative (persoana intai)
|
||||
"Mi-am facut cont, am cheia, trimit prima prestatie — merge (sunt in trial). Construiesc integrarea,
|
||||
folosesc `valideaza` ca sa testez fara sa consum nimic — perfect. Peste o luna, trial-ul expira; brusc
|
||||
`POST /v1/prezentari` da 403. Daca mesajul zice doar '403 Forbidden', cred ca mi-am stricat cheia si pierd
|
||||
o ora. Daca zice 'Importul prin API e pe planul Pro — scrie-ne la X ca sa activam', stiu exact ce sa fac."
|
||||
|
||||
### DX Scorecard (8 dimensiuni, 0-10)
|
||||
```
|
||||
1. TTHW 7 (free->403 dupa trial surprinde fara mesaj clar)
|
||||
2. Naming consistency 9
|
||||
3. Error actionability 6 -> tinta 9 dupa copy fix
|
||||
4. Docs/exemple 6 -> documenteaza valideaza-pe-free + upgrade
|
||||
5. Progressive disclosure 8 (nomenclator+valideaza publice/permise)
|
||||
6. Escape hatches 7 (dev id=1; flag enforcement)
|
||||
7. Upgrade safety 6 (manual; link concret lipseste)
|
||||
8. Consistency cross-canal 8
|
||||
--------------------------------
|
||||
Overall DX ~7.1/10 (TTHW: ~5 min ramane; tinta erori/docs ~8.5)
|
||||
```
|
||||
|
||||
### DX Implementation Checklist
|
||||
- [ ] `PLAN_FARA_API`: fix actionabil cu canal de contact concret (email/telefon), mentioneaza `valideaza`.
|
||||
- [ ] `PLAN_LIMITA_LUNARA`: fix cu "mai poti trimite N luna asta" + cum treci pe alt plan.
|
||||
- [ ] Doc scurt pentru integratori: "dezvolta pe free cu `valideaza`; trimiterea reala cere Pro".
|
||||
- [ ] `set-tier` help text clar (CLI) + audit in app_events.
|
||||
- [ ] Confirma `valideaza` ramane permis pe orice plan (decizie -> default PERMIS).
|
||||
|
||||
**Phase 3.5 complete.** DX overall ~7.1/10. TTHW ~5 min (neschimbat pentru cont cu drept). Codex:
|
||||
indisponibil. Claude subagent: 3 constatari (1 HIGH mesaj-403, 2 MEDIUM docs/upgrade-link) + leaga
|
||||
T-CEO-3 (valideaza gated vs permis). Trec la Faza 4.
|
||||
|
||||
<!-- AUTONOMOUS DECISION LOG -->
|
||||
## Decision Audit Trail
|
||||
|
||||
| # | Faza | Decizie | Clasificare | Principiu | Rationament | Respins |
|
||||
|---|------|---------|-------------|-----------|-------------|---------|
|
||||
| 1 | CEO | Mod = SELECTIVE EXPANSION | Mechanical | override autoplan | iteratie pe sistem existent | EXPANSION/HOLD/REDUCTION |
|
||||
| 2 | CEO | `effective_tier(acct, now)` cu `now` injectabil | Mechanical | P5 explicit | teste de granita deterministe | now intern (flaky) |
|
||||
| 3 | CEO | Coduri noi ca erori business 3-niveluri (nu 500) | Mechanical | P4 DRY/errors.py | reuse pattern existent | exceptii/catch-all |
|
||||
| 4 | CEO | Gate volum/API server-side dupa resolve_account_id | Mechanical | P1 completeness/sec | nu pe camp din body | gate din body (nesigur) |
|
||||
| 5 | CEO | Accepta overshoot mic cota sub concurenta | Taste->auto | P3 pragmatic | lock per-cont = over-eng pt cap soft | lock tranzactional |
|
||||
| 6 | CEO | Valoarea 60 + capability EXCLUSIV in PLANS | Mechanical | P4 DRY | o singura sursa | hardcodare in 3 locuri |
|
||||
| 7 | CEO | log_event pe fiecare respingere de plan | Mechanical | zero-silent-failures | "de ce blocat X?" vizibil | doar 4xx tacut |
|
||||
| 8 | CEO | Index `(account_id,created_at)` deferat -> TODO | Mechanical | P3 | volume mici acum | index acum (premature) |
|
||||
| 9 | CEO | T-CEO-1: enforcement sub flag + soft-first volum | **USER CHALLENGE -> REZOLVAT** | decizie user (2026-06-28) | **enforcement DUR direct de la deploy**; fara conturi legacy, pre-productie -> riscul de fals-block e moot | soft-first / flag-OFF respinse |
|
||||
| 10 | CEO | T-CEO-2: limita 60 ca o constanta config | Taste | P5 | tunabila fara cod | hardcodat |
|
||||
| 11 | Design | 6 stari explicite afisaj in US-006 | Mechanical | P1 completeness | acoperire stari | doar 3 stari |
|
||||
| 12 | Design | Copy RO fix cu pluralizare (zi/zile) | Mechanical | P5 explicit | nu lasa implementatorului | generic |
|
||||
| 13 | Design | T-DES-1: banner one-time la trial->free | Taste | P1 | semnal la trecere | doar badge tacut |
|
||||
| 14 | Design | warn = culoare + text (nu doar culoare) | Mechanical | P1 a11y | accesibilitate | doar culoare |
|
||||
| 15 | Eng | `CONSUMED_STATUSES` constanta exportata | Mechanical | P4 DRY | nu duplica definitia consum | duplicare in teste |
|
||||
| 16 | Eng | Fara tabela `plan_usage` (foloseste created_at) | Mechanical | P3/non-goal | migrare minima | coloana/tabela noua |
|
||||
| 17 | Eng | 2 invariante critice ca teste (retry, dry-run) | Mechanical | P1 completeness | usor de stricat | a le omite |
|
||||
| 18 | DX | `valideaza` ramane PERMIS pe orice plan (default) | Taste->auto | P1 DX | dezvolti pe free, trimiti pe Pro | gated ca restul API |
|
||||
| 19 | DX | Fix erori plan cu canal de contact concret | Mechanical | P1 completeness | actionabil | "contacteaza-ne" vag |
|
||||
| 20 | All | "prestatie consumata" = queued+sending+sent | Taste->auto | P1 | limita pe ce trimitem la RAR | doar sent |
|
||||
| 21 | All | Lot peste limita -> respingere totala clara | Taste->auto | P5 explicit | evita surprize enqueue partial | partial tacut |
|
||||
| 22 | All | **Enforcement DUR direct de la deploy** (rezolva T-CEO-1) | **USER DECISION (2026-06-28)** | user-stated | fara conturi legacy, produs in TESTE/pre-productie -> riscul de fals-block e moot; flag = optional kill-switch | soft-first / flag-OFF |
|
||||
| 23 | CEO | **T-CEO-2 REZOLVAT: limita 60 = constanta config tunabila** (o singura sursa in plans.py/config) | **USER DECISION (2026-06-28)** | user-stated (pe recomandare) | DRY/tunabil fara arheologie de cod | hardcodat |
|
||||
| 24 | Design | **T-DES-1 REZOLVAT: banner one-time la expirarea trial->Gratuit** | **USER DECISION (2026-06-28)** | user-stated (pe recomandare) | semnal clar la trecere, evita surpriza la prima respingere | doar badge |
|
||||
| 25 | DX | **T-DX-3 REZOLVAT: `valideaza` dry-run ramane PERMIS pe orice plan** | **USER DECISION (2026-06-28)** | user-stated (pe recomandare) | dezvolti pe free, trimiti pe Pro — DX excelent | gated ca restul API |
|
||||
|
||||
## Cross-Phase Themes
|
||||
- **Tema: enforcement fara cale de conversie** — semnalata in CEO (S9/S10) + DX (upgrade path). Semnal
|
||||
inalt: hard-block + lipsa self-service = friction. -> sustine T-CEO-1.
|
||||
- **Tema: mesaje oneste, actionabile** — CEO (S2/S8) + Design (P4 copy) + DX (erori). Convergent:
|
||||
fiecare respingere are problema+cauza+fix + canal de contact.
|
||||
- **Tema: determinism temporal** — CEO (S1 now injectabil) + Eng (S3 teste granita) + Design (pluralizare
|
||||
zile). `now` injectabil + timp local RO sunt fundatia testelor.
|
||||
|
||||
## TODOS.md (propuneri)
|
||||
- **[P3] Index `(account_id, created_at)` pe submissions** — cand apar conturi cu mii prestatii/luna,
|
||||
`monthly_usage` scaneaza randurile lunii. Efort S. Depinde de: aparitia volumului mare. (A: adauga la TODOS)
|
||||
- **[P2] Job eager downgrade `trial_until` expirat -> NULL** — igiena in purjarea orara T16; lazy acopera
|
||||
corectitudinea. Efort S. (A: adauga la TODOS, optional)
|
||||
- **[P1->Phase 2] Billing self-service (upgrade din UI)** — golul strategic; fara el enforcement-ul produce
|
||||
churn in loc de conversie. Efort XL. PRD separat. (A: adauga la TODOS ca Phase 2)
|
||||
- **[P3] Re-trial / nurture la expirare** — email "trial expirat, treci pe Pro". Efort M. (A: TODOS)
|
||||
|
||||
## Implementation Tasks (sintetizate)
|
||||
- [ ] **T1 (P1, human ~3h / CC ~25min) — schema/plans** — `accounts.tier`+`trial_until` (migrare aditiva
|
||||
defensiva) + `app/plans.py` (PLANS, `effective_tier(acct,now)`, `monthly_usage(conn,acct,now)`,
|
||||
`CONSUMED_STATUSES`). Surfaced by: CEO S1 / Eng S1-S2. Files: schema.sql, db.py, app/plans.py, accounts.py.
|
||||
Verify: test_migrare_*, test_plan_definitii, test_effective_tier_*.
|
||||
- [ ] **T2 (P1, human ~2h / CC ~15min) — accounts** — `create_account` seteaza trial Pro 30z; `set_tier`
|
||||
(protejat id=1); legacy -> free fara trial. Surfaced by: CEO 0B / Eng. Files: accounts.py, tools/account.py.
|
||||
- [ ] **T3 (P1, human ~3h / CC ~25min) — enforce volum** — gate INAINTE de build_key pe ambele canale +
|
||||
cod `PLAN_LIMITA_LUNARA` + log_event; lot peste limita -> respingere totala. Surfaced by: CEO S3/S4/S8.
|
||||
Files: api/v1/router.py, import_router.py, web/routes.py, errors.py. Verify: test_free_peste_60_*, retry.
|
||||
- [ ] **T4 (P1, human ~2h / CC ~15min) — gate API** — `require_api_access` (Pro+/trial) pe rutele de
|
||||
ingestie API; `valideaza`+`nomenclator` raman permise; dev id=1 exceptat; cod `PLAN_FARA_API` (403 actionabil).
|
||||
Files: auth.py, api/v1/router.py, import_router.py, errors.py. Verify: test_*_api_403/ok.
|
||||
- [ ] **T5 (P3 OPTIONAL, human ~30min / CC ~5min) — flag enforcement (kill-switch)** — `AUTOPASS_ENFORCE_PLANS`
|
||||
(config). NU blocant: enforcement DUR e activ implicit de la deploy (decizie user). Flag-ul = doar
|
||||
comoditate de operare. Files: config.py + gate-urile. Surfaced by: CEO S9 (rezolvat).
|
||||
- [ ] **T6 (P2, human ~3h / CC ~20min) — UI dashboard** — badge plan antet + linie burger + consum N/60 +
|
||||
warn>=80% + 6 stari + copy RO pluralizat + pagina Cont. Surfaced by: Design P2/P4. Files: web/routes.py,
|
||||
templates/_status.html,_cont.html. Verify: test_afisaj_*, test_copy_pluralizare.
|
||||
- [ ] **T7 (P1, human ~30min / CC ~5min) — landing copy** — 100->60 (linii 7,65,92,266,388);
|
||||
"Premium gratuit 30 zile"->"Pro gratuit 30 zile" (256,350). Files: landing.html. Verify: test_landing_*.
|
||||
- [ ] **T8 (P2, human ~1h / CC ~10min) — teste matrice E2E** — plan x capabilitate x canal x trial +
|
||||
granita luna RO + dev id=1. Files: tests/test_plans.py, test_api_scope.py, test_web_*. Verify: pytest -q.
|
||||
- [ ] **T9 (P2, human ~30min / CC ~5min) — docs integrator** — "dezvolta pe free cu valideaza, trimiterea
|
||||
reala cere Pro". Surfaced by: DX. Files: docs/ + integrare_examples.
|
||||
|
||||
## GSTACK REVIEW REPORT
|
||||
|
||||
| Review | Trigger | Why | Runs | Status | Findings |
|
||||
|--------|---------|-----|------|--------|----------|
|
||||
| CEO Review | `/plan-ceo-review` | Scop & strategie | 1 | issues_open | 9 constatari, CRITICAL GAP legacy REZOLVAT (moot), mode SELECTIVE_EXPANSION |
|
||||
| Codex Review | `/codex review` | A 2-a opinie | 0 | indisponibil | limita utilizare (pana 2026-07-18) |
|
||||
| Eng Review | `/plan-eng-review` | Arhitectura & teste (required) | 1 | issues_open | 4 constatari, gap legacy moot, 2 invariante critice teste |
|
||||
| Design Review | `/plan-design-review` | UI/UX | 1 | issues_open | 4 constatari, overall ~6.4/10 |
|
||||
| DX Review | `/plan-devex-review` | Developer experience | 1 | issues_open | 3 constatari, DX ~7.1/10 |
|
||||
|
||||
- **VERDICT:** CEO + Design + Eng + DX rulate (subagent-only, codex indisponibil). Toate deciziile inchise
|
||||
(2026-06-28): USER CHALLENGE rezolvat (enforcement DUR direct de la deploy; CRITICAL GAP migrare = moot,
|
||||
fara conturi legacy/pre-productie) + cele 3 taste decisions rezolvate pe recomandare (T-CEO-2 constanta
|
||||
config, T-DES-1 banner one-time trial->Gratuit, T-DX-3 `valideaza` permis pe orice plan). Plan gata de executie.
|
||||
|
||||
NO UNRESOLVED DECISIONS
|
||||
355
docs/prd/prd-5.18-corpus-knn-exemple-etichetate.md
Normal file
355
docs/prd/prd-5.18-corpus-knn-exemple-etichetate.md
Normal file
@@ -0,0 +1,355 @@
|
||||
# PRD 5.18 — Corpus k-NN din exemple reale etichetate (mapare operatii service)
|
||||
|
||||
**Stare**: aprobat + revizuit /autoplan (2026-06-28; intrebari deschise rezolvate de user — vezi §5 Decizii;
|
||||
cerinte user D4/D5 + 10 constatari Eng incorporate — vezi GSTACK REVIEW REPORT la final)
|
||||
|
||||
> Proces: `docs/ROADMAP.md` §5. Contract RAR: `docs/api-rar-contract.md`. Construieste peste
|
||||
> infrastructura 5.14 (straturi GOLD/SILVER/embeddings, `app/embeddings.py`, `app/shared_store.py`,
|
||||
> `mapping_suggestions`). NU re-deschide deciziile 5.14 (#11-#19); le foloseste.
|
||||
|
||||
## 0. Context si motivatie (de ce acest PRD)
|
||||
|
||||
5.14 a livrat embeddings in-proces, dar corpusul indexat = **cele 18 denumiri RAR generice**
|
||||
din nomenclator (`nume_prestatie` -> `cod_prestatie`). O operatie reala ("inlocuit lubrifiant
|
||||
la propulsor") se potriveste semantic slab cu etichete generice scurte ("INTRETINERE",
|
||||
"REPARATIE"). In plus, stratul **SILVER (`mapping_suggestions`) e populat DOAR in teste** —
|
||||
in productie e gol, deci nu produce nicio sugestie (LLM-ul nu e chemat la runtime).
|
||||
|
||||
Acest PRD muta corpusul de la cele 18 categorii la **operatiile reale etichetate** (k-NN peste
|
||||
exemple): o operatie noua se potriveste semantic cu o operatie deja vazuta si MOSTENESTE codul ei.
|
||||
|
||||
**Masuratori care justifica directia** (vezi memorie `test-precizie-knn-embeddings`, rulat 2026-06-28):
|
||||
- k-NN peste exemple etichetate: **94.3% acord cu LLM pe operatii distincte** (baseline "mereu OE-1" = 86.2%).
|
||||
- Acoperire IEFTINA: pe volumul real total (155.195 aparitii, 17.181 operatii distincte):
|
||||
148 operatii = 50% volum, **1.380 = 80%**, 4.368 = 90%, 9.422 = 95%.
|
||||
- Punct slab masurat: **NUL recall 64%** (ITP/discount/plata scapa ca OE-1) -> de aici pre-filtrul (US-001).
|
||||
- Etichetarea offline cu **Qwen3-4B local (LM Studio, GPU RX 6600M)** + prompt procedural in 3 pasi:
|
||||
**91% pe batch greu, 20/20 pe batch de validare**, ambele NUL prinse. Debit ~1.5-2h pentru ~13.5k operatii.
|
||||
|
||||
## 1. Obiectiv
|
||||
|
||||
Inlocuieste corpusul embeddings (18 categorii generice) cu **corpusul de operatii reale etichetate**
|
||||
(exemplu -> cod RAR), populat dintr-un seed comis in repo, plus un **pre-filtru determinist** pentru
|
||||
non-operatii (NUL). Rezultat: sugestii de mapare semnificativ mai precise in editor, fara LLM la runtime.
|
||||
|
||||
**Pasul 1 (bootstrap offline, fundatia intregului PRD) = etichetare cu LLM via LM Studio local.**
|
||||
Tot restul (seeder, corpus embeddings, enrich) consuma artefactul produs aici. Pasul are doua garantii
|
||||
non-negociabile:
|
||||
1. **LM Studio = backend implicit aprobat pentru rularea v1** (Qwen3-4B local, GPU RX 6600M, `json_schema`
|
||||
strict — `json_object` e respins de LM Studio). Groq/OpenRouter raman fallback-uri interschimbabile, dar
|
||||
NU sunt calea aprobata pentru bootstrap-ul v1 (vezi D4).
|
||||
2. **Dedup INAINTE de orice apel LLM.** Cele 4 fisiere (`docs/operatii-service/*.csv`) contin **19.456 randuri
|
||||
brute -> 17.181 operatii distincte dupa `normalize_for_match`** (gain de doar 254 fata de dedup exact-string,
|
||||
pentru ca datele sunt deja majuscule, fara diacritice — `normalize_for_match` colapseaza spatii + scoate diacritice,
|
||||
**NU** scoate punctuatie). Din cele 17.181, **3.662 sunt deja etichetate** (in spatiu normalizat) in
|
||||
`labels-groq-partial.json`. Trimitem la LLM EXACT cele **13.519** operatii distincte ne-etichetate, niciodata un
|
||||
duplicat normalizat, o cheie normalizata vida sau o operatie deja etichetata (vezi D5). Economie: **31% mai putine
|
||||
apeluri** vs randuri brute. (Castigul real al pipeline-ului nu e atat normalizarea — 254 chei — cat **reuse-ul
|
||||
etichetelor existente** + agregarea frecventei; motivul principal pentru spatiul normalizat e **consistenta
|
||||
end-to-end cu cheia DB/k-NN**, vezi F1/F3 din review.)
|
||||
|
||||
## 2. Non-Goals (anti scope-creep)
|
||||
|
||||
- **NU auto-send peste GOLD propriu.** Toate sursele (k-NN, exact, NUL pre-filtru) raman SUGGESTION-ONLY,
|
||||
niciodata in `resolve_prestatii`/`load_mapping` (invariant #13, #11 din 5.14). Singura cale spre `queued`
|
||||
ramane `operations_mapping` (GOLD propriu confirmat de om).
|
||||
- **NU LLM la runtime.** Etichetarea LLM se face O SINGURA DATA, offline; runtime = doar embeddings + exact + reguli.
|
||||
- **NU validare temporala / re-etichetare automata.** Seedul e static; reimprospatarea e un re-run manual al tool-ului.
|
||||
- **NU schimbare UI majora.** Editorul (`_mapari.html`) consuma deja `sugestie_principala`; doar sursa se schimba.
|
||||
(Un badge optional de sursa = US-007, jos.)
|
||||
- **NU eshantion etichetat de om in acest PRD** (doar mentionat la Riscuri ca recomandare — Decision #19).
|
||||
|
||||
## 3. Stories atomice
|
||||
|
||||
> Fiecare story = cea mai mica unitate care lasa sistemul functional. Refoloseste `mapping_suggestions`
|
||||
> (SILVER) ca tabela-corpus (are deja: `denumire_normalizata`, `cod_prestatie`, `is_nul`, `source`,
|
||||
> `confidence`) — populata acum si in productie, nu doar in teste.
|
||||
|
||||
### US-001: Pre-filtru determinist non-operatii (NUL)
|
||||
**Ca** operator **vreau** ca gunoiul evident (ITP, plata, discount, nr. inmatriculare, tractare) sa fie
|
||||
marcat NUL inainte de k-NN **pentru ca** masuratoarea arata recall NUL doar 64% (scapa ca OE-1).
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `app/mapping.py` (functie noua `prefiltru_nul(denumire) -> bool`), `tests/test_prefiltru_nul.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_prefiltru_nul.py` — `test_itp_e_nul`, `test_plata_discount_nul`, `test_nr_inmatriculare_nul`, `test_operatie_reala_nu_e_nul`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Reguli text/regex deterministe (ITP, ACHITAT/PLATA, DISCOUNT/REDUCERE, NR INMATRICULARE + pattern placuta, TRACTARE, TAXA)
|
||||
- [ ] `prefiltru_nul("13 X ITP")` / `("DISCOUNT FIDELITATE 10%")` -> True; `("INLOCUIT PLACUTE FRANA")` -> False
|
||||
- [ ] Zero fals-pozitiv pe un set de 20 operatii reale (din `docs/operatii-service`)
|
||||
- [ ] `python3 -m pytest tests/test_prefiltru_nul.py -q` verde
|
||||
- **Verificare E2E**: — (pur backend, acoperit de teste)
|
||||
|
||||
### US-002: Etichetator offline multi-backend cu prompt procedural
|
||||
**Ca** dezvoltator **vreau** un tool care eticheteaza operatii->coduri RAR via LM Studio local / Groq /
|
||||
OpenRouter, cu prompt procedural in 3 pasi si `json_schema` strict **pentru ca** LM Studio respinge
|
||||
`json_object` si promptul nou ridica precizia (91% vs 80%).
|
||||
|
||||
- **Depinde de**: —
|
||||
- **Fisiere**: `tools/mapare-llm/eticheteaza.py` (NOU, backend-uri interschimbabile), `tests/test_eticheteaza_tool.py` (mock HTTP) (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_eticheteaza_tool.py` — `test_construieste_prompt_3pasi`, `test_parseaza_json_schema`, `test_backend_selectabil_env`, `test_scrub_pii_inainte_de_request`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Backend selectabil prin env (`ETICHETARE_BACKEND=lmstudio|groq|openrouter`, endpoint+model configurabile);
|
||||
**default = `lmstudio`** (backend-ul aprobat pentru bootstrap v1, D4). Groq/OpenRouter = fallback.
|
||||
- [ ] `response_format` = `json_schema` strict cu **envelope complet** `{"type":"json_schema","json_schema":{"name":...,"strict":true,"schema":{...}}}`
|
||||
(NU `{"type":"json_object"}` ca `or_common.py:57`/`label_common.py:24`); `cod` = **enum** peste cele 19 `ALL_LABELS` (18 + NUL),
|
||||
cod invalid/lipsa -> `?` (F8 din review). Etichetatorul nou NU reutilizeaza request-ul vechi, doar promptul/codurile/scrub-ul.
|
||||
- [ ] **Dezactiveaza explicit "thinking"-ul Qwen3** (`/no_think` sau reasoning off) — altfel modelul emite `<think>` si
|
||||
umfla tokeni/latenta sub structured output strict (F8).
|
||||
- [ ] **Garda de truncare**: daca raspunsul are mai putine iteme decat batch-ul sau JSON invalid -> log + marcheaza `?`
|
||||
pe pozitiile lipsa, NU le ascunde tacit (la batch 40 + prompt 3 pasi, `n_ctx=4096` e stramt — F8).
|
||||
- [ ] Promptul = procedura 3 pasi + ancore (mapare parte caroserie->OE-C etc.), versionat in fisier
|
||||
- [ ] Scrub PII (nr. inmatriculare, VIN) inainte de orice request (refoloseste `or_common.scrub`, #3)
|
||||
- [ ] Setari conservatoare documentate in tool (batch 32-40, `n_parallel=1`, `n_ctx=4096`) — vezi Riscuri
|
||||
- [ ] `python3 -m pytest tests/test_eticheteaza_tool.py -q` verde (fara retea reala)
|
||||
- **Verificare E2E**: rulare manuala 1 batch pe LM Studio local (`http://<tailscale>:1234`), confirmare JSON valid
|
||||
|
||||
### US-003: Generare seed etichetat in faze pe frecventa
|
||||
**Ca** dezvoltator **vreau** sa generez un fisier seed `operatii-etichetate.json` (operatie->cod) pornind de la
|
||||
operatiile existente + cele deja etichetate, in ordinea frecventei **pentru ca** 1.380 operatii prind 80% din volum.
|
||||
|
||||
- **Depinde de**: US-002
|
||||
- **Fisiere**: `tools/mapare-llm/genereaza_seed.py` (NOU), `app/data/operatii-etichetate.json` (artefact comis), `tests/test_genereaza_seed.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_genereaza_seed.py` — `test_dedup_normalizat`, `test_zero_duplicate_trimis_la_llm`, `test_rerun_zero_apeluri_llm`, `test_reuse_conflict_determinist`, `test_skip_cheie_normalizata_vida`, `test_reuse_in_spatiu_normalizat`, `test_ordine_pe_frecventa`, `test_format_seed_valid`
|
||||
- **Pipeline dedup (ordinea e obligatorie, INAINTE de orice apel LLM):**
|
||||
1. Agrega cele 4 CSV-uri -> pentru fiecare rand `(denumire, NR)`. Parseaza NR tolerant (skip rand pe NR ne-numeric, nu zero-weight — F9).
|
||||
2. `cheie = normalize_for_match(denumire)` — ACEEASI functie ca DB/k-NN (`app/mapping.py:40`), NU `.strip()` exact.
|
||||
**Arunca randurile cu `cheie == ""`** (gunoi gen `"..."`, `" "`) inainte de dedup — altfel se bat pe slotul UNIQUE gol (F6).
|
||||
3. Dedup pe cheie: un singur reprezentant per cheie, `freq = suma NR` pe toate aparitiile/fisierele.
|
||||
4. Construieste **harta** `cheie_normalizata -> cod` (NU doar un set) din TOATE sursele de etichete deja existente:
|
||||
`labels-groq-partial.json` (cheiat pe text BRUT) **PLUS seedul comis anterior** `operatii-etichetate.json` (cheiat normalizat).
|
||||
Reuse + scaderea se fac in spatiu normalizat. **Rezolvare conflict determinista** cand acelasi `cheie` are coduri diferite
|
||||
pe variante raw (masurat: 1 azi — `CURATAT CATALIZATOR` OE-2 vs OE-1): castiga varianta cu `freq` (suma NR) maxima, tie-break pe `cod` sortat (F3).
|
||||
5. `de_etichetat = {cheie in corpus} - {cheie in harta etichete}`. Lista (distincta, ne-etichetata, sortata desc pe freq) = SINGURUL input catre LLM.
|
||||
- **Acceptance criteria**:
|
||||
- [ ] `test_zero_duplicate_trimis_la_llm` (within-run): backend LLM mock care inregistreaza fiecare denumire primita;
|
||||
input cu duplicate intentionate (spatii/case + cross-file) -> mock-ul nu vede NICIODATA doua chei normalizate egale,
|
||||
nicio cheie deja etichetata, nicio cheie vida.
|
||||
- [ ] `test_rerun_zero_apeluri_llm` (cross-run, **criteriul real de idempotenta**, F2/F7): ruleaza tool-ul de doua ori cu acelasi
|
||||
input; a doua rulare consuma seedul comis ca cache -> **0 apeluri LLM**, seed identic byte-cu-byte.
|
||||
- [ ] `test_reuse_conflict_determinist` (F3/F7): doua variante raw ale aceleiasi chei cu coduri diferite -> codul ales e determinist (freq-max, tie-break cod).
|
||||
- [ ] Dedup pe `normalize_for_match` (colapseaza spatii + diacritice, **NU** punctuatie; gain real ~254 chei vs exact-string —
|
||||
valoarea principala e consistenta cu cheia DB/k-NN, nu volumul); NU reutiliza `or_common.corpus_by_freq()` ca atare (dedup exact-string).
|
||||
- [ ] Eticheteaza DOAR ce lipseste, in ordine descrescatoare de frecventa, cu `--target-volum 0.9` (oprire la prag) sau `--all`
|
||||
- [ ] Seed format `[{denumire, denumire_normalizata, cod, is_nul, source, confidence}]`, UTF-8, comis in repo;
|
||||
`denumire_normalizata` unica + ne-vida in seed (oglindeste UNIQUE din `mapping_suggestions`; `test_format_seed_valid` asserta non-empty)
|
||||
- [ ] `python3 -m pytest tests/test_genereaza_seed.py -q` verde
|
||||
- **Verificare E2E**: rulare `--target-volum 0.5` pe date reale -> ~150 etichete noi, fisier valid; log-ul tool-ului
|
||||
raporteaza explicit "{brute} randuri -> {distincte} dupa normalizare -> {de_etichetat} trimise la LLM"
|
||||
|
||||
### US-004: Seeder corpus etichetat in DB (mapping_suggestions)
|
||||
**Ca** sistem **vreau** sa incarc seedul etichetat in `mapping_suggestions` la init (INSERT OR IGNORE)
|
||||
**pentru ca** SILVER e gol in productie si trebuie populat ca sa dea sugestii exact-match + corpus k-NN.
|
||||
|
||||
- **Depinde de**: US-003
|
||||
- **Fisiere**: `app/operatii_seed.py` (NOU, dupa modelul `nomenclator_seed.py`), `app/db.py` (apel la init), `tests/test_operatii_seed.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_operatii_seed.py` — `test_seed_populeaza_mapping_suggestions`, `test_insert_or_ignore_nu_clobber_uman`, `test_is_nul_din_seed`, `test_idempotent_la_reinit`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] La `init_db`, daca seedul exista si tabela permite, INSERT OR IGNORE randurile (idempotenta re-seed: nu dubla / nu
|
||||
clobber un rand seedat sau de embedding deja prezent). NB (F10): confirmarile UMANE stau in `shared_mappings`
|
||||
(`record_human_validation`), NU in `mapping_suggestions` — deci INSERT OR IGNORE pastreaza TACIT codul LLM vechi la re-seed;
|
||||
daca vrei refresh pe coduri LLM invechite, e decizie explicita upsert-vs-ignore (v1 = ignore)
|
||||
- [ ] `is_nul=1` -> `cod_prestatie=NULL` (respecta CHECK-ul existent); `source='llm_seed'`, `confidence` din seed
|
||||
- [ ] Idempotent: a doua initializare nu dubleaza si nu modifica randuri existente
|
||||
- [ ] `python3 -m pytest tests/test_operatii_seed.py -q` verde
|
||||
- **Verificare E2E**: pornire app pe DB gol -> `SELECT count(*) FROM mapping_suggestions` > 0
|
||||
|
||||
### US-005: Embeddings indexeaza corpusul etichetat (nu nomenclatorul)
|
||||
**Ca** sistem **vreau** ca `ensure_embeddings_corpus` sa indexeze operatiile etichetate (denumire->cod, cu is_nul)
|
||||
**pentru ca** k-NN peste exemple reale e net mai precis decat peste 18 categorii generice.
|
||||
|
||||
- **Depinde de**: US-004
|
||||
- **Fisiere**: `app/mapping.py` (`ensure_embeddings_corpus` schimba sursa), `app/embeddings.py` (`suggest_nearest` intoarce si `is_nul`), `tests/test_embeddings_corpus_etichetat.py` (~3 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_embeddings_corpus_etichetat.py` — `test_corpus_din_mapping_suggestions`, `test_suggest_nearest_intoarce_is_nul`, `test_semnatura_corpus_pe_seed`, `test_degradare_gratioasa_pastrata`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Corpusul = `mapping_suggestions` (denumire_normalizata -> cod, is_nul), NU `nomenclator_rar`
|
||||
- [ ] **Simetrie corpus/query (F1, HIGH)**: corpusul e text `denumire_normalizata`; deci `enrich_suggestions` trebuie
|
||||
sa interogheze `suggest_nearest(normalize_for_match(denumire), ...)`, NU `denumire` brut. Altfel corpus normalizat vs
|
||||
query brut degradeaza cosine si NU e configul sub care s-a masurat 94.3%. `test_query_normalizat_ca_si_corpusul` o asserta.
|
||||
- [ ] `suggest_nearest` intoarce `[{cod, is_nul, similaritate}]`; un vecin NUL -> semnal de supresie, nu cod
|
||||
- [ ] Re-index doar la schimbarea semnaturii corpusului (cache pastrat, #16b degradare gratioasa neschimbata)
|
||||
- [ ] Gated pe `AUTOPASS_EMBEDDINGS_ENABLED` (acum default True — vezi 5.14 CLOSE); off in teste (conftest)
|
||||
- [ ] `python3 -m pytest tests/test_embeddings_corpus_etichetat.py -q` verde
|
||||
- **Verificare E2E**: cu flag on + seed incarcat, `suggest_nearest("schimbat uleiul motor")` -> cod revizie/intretinere real
|
||||
|
||||
### US-006: enrich_suggestions = pre-filtru NUL + k-NN pe corpus etichetat
|
||||
**Ca** operator **vreau** ca editorul sa imbine pre-filtrul NUL, exact-match si k-NN semantic in ordinea de
|
||||
precedenta corecta **pentru ca** vreau sugestia cea mai buna fara junk.
|
||||
|
||||
- **Depinde de**: US-001, US-005
|
||||
- **Fisiere**: `app/mapping.py` (`enrich_suggestions`), `tests/test_enrich_corpus_etichetat.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_enrich_corpus_etichetat.py` — `test_prefiltru_nul_supreseaza_inainte_de_knn`, `test_precedenta_gold_exact_embedding`, `test_prag_similaritate`, `test_abtinere_sub_prag`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Ordine: pre-filtru NUL -> daca NUL, fara sugestie de cod (marcat non-operatie); altfel GOLD partajat > exact (SILVER) > k-NN embeddings
|
||||
- [ ] k-NN sub `EMB_MIN_SIMILARITATE` -> abtinere (`embedding=None`), nu sugestie incerta
|
||||
- [ ] Vecin k-NN cu `is_nul=1` -> tratat ca supresie, nu cod (consecventa cu pre-filtrul)
|
||||
- [ ] Invariant #13 pastrat: nimic din asta nu intra in `resolve_prestatii`/`load_mapping` (test de regresie)
|
||||
- [ ] `python3 -m pytest tests/test_enrich_corpus_etichetat.py -q` verde + suita 5.14 (`test_mapare_integrare_l14.py`) ramane verde
|
||||
- **Verificare E2E**: browser HTMX pe `/_fragments/mapari` — operatie parafraza primeste cod corect pre-selectat din k-NN
|
||||
|
||||
### US-007 (optional): Badge sursa sugestie in editor
|
||||
**Ca** operator **vreau** sa vad de unde vine sugestia (confirmat de om / exemplu similar / non-operatie)
|
||||
**pentru ca** acum nu pot distinge sursa si nu stiu cata incredere sa am.
|
||||
|
||||
- **Depinde de**: US-006
|
||||
- **Fisiere**: `app/web/templates/_mapari.html`, `tests/test_web_badge_sursa.py` (~2 fisiere)
|
||||
- **Test intai (RED)**: `tests/test_web_badge_sursa.py` — `test_badge_gold`, `test_badge_embedding`, `test_badge_nul`
|
||||
- **Acceptance criteria**:
|
||||
- [ ] Chip mic langa sugestie: "confirmat" (gold), "similar" (embedding/silver), "non-operatie" (NUL)
|
||||
- [ ] Fara sursa -> fara chip; nu rupe layoutul 5.15/5.16
|
||||
- [ ] `python3 -m pytest tests/test_web_badge_sursa.py -q` verde
|
||||
- **Verificare E2E**: browser — chip vizibil si corect colorat pe randul de mapare
|
||||
|
||||
## 4. Riscuri
|
||||
|
||||
- **Calitate etichetare model local (Qwen3-4B Q4) < model mare (Groq 70b).** Masurat: bun pe cap (frecvent,
|
||||
clar), mai slab pe coada rara/ambigua (ADAS calibrare, chei, "doar nume piesa"). Mitigare: pre-filtru NUL
|
||||
(US-001) + optiunea unui al doilea pas de verificare cloud DOAR pe esantionul cu cod rar/incert.
|
||||
- **Hardware GPU-box instabil sub sarcina (shutdown observat 2026-06-29).** La config-ul rulant erau ~4GB VRAM
|
||||
liberi -> cauza probabil termica/alimentare, NU memorie. Mitigare OBLIGATORIE pentru pasul de etichetare:
|
||||
`n_parallel=1`, `n_ctx=4096`, batch 32-40, monitorizare temperatura GPU. NU mari batch/context fara headroom termic.
|
||||
- **Ground-truth = eticheta LLM, nu om.** 94.3% e ACORD cu LLM, nu acuratete reala; LLM impinge 86% in OE-1
|
||||
(posibil prea agresiv). **Recomandare (Decision #19):** inainte de a creste increderea/orice auto-send, ruleaza
|
||||
`heldout_eval.py` cu un esantion etichetat de OM. Ramane in afara scope-ului acestui PRD, dar e poarta pentru orice 5.x viitor de auto-send.
|
||||
- **`mapping_suggestions` populat schimba comportamentul testelor** care presupuneau SILVER gol. Mitigare: seederul
|
||||
ruleaza doar daca seedul exista; conftest poate dezactiva seedul in testele care nu-l vor (ca la embeddings).
|
||||
- **Coada lunga ramane needs_mapping.** Chiar la 90% volum acoperit, 76% din operatiile DISTINCTE raman neetichetate
|
||||
(frecventa 1). Asteptare corecta: bootstrap-ul reduce mult volumul, dar editorul uman ramane necesar pe coada.
|
||||
- **(F1, review) Simetrie corpus/query la embeddings.** Corpusul k-NN devine text NORMALIZAT (`denumire_normalizata`),
|
||||
deci query-ul TREBUIE normalizat la fel inainte de embedding (US-005 AC). Daca raman asimetrice (corpus normalizat,
|
||||
query brut), similaritatea scade si nu mai e configul masurat (94.3%). Risc de regresie tacuta — acoperit de test in US-005.
|
||||
- **(F2, review) Idempotenta cross-run a etichetarii.** Etichetele noi produse de o rulare trebuie sa devina cache pentru
|
||||
urmatoarea (seedul comis = sursa de etichete, nu doar `labels-groq-partial.json`), altfel re-run-ul re-trimite tot la LLM.
|
||||
Acoperit de `test_rerun_zero_apeluri_llm` (US-003).
|
||||
|
||||
## 5. Decizii (intrebari deschise rezolvate la aprobare, 2026-06-28)
|
||||
|
||||
> Erau intrebari deschise; rezolvate de user la poarta de aprobare PRD. Devin constrangeri de executie.
|
||||
|
||||
- **D1 — Tinta de acoperire la etichetare: 90% din volum** (`--target-volum 0.9`, ~4.368 operatii distincte).
|
||||
Restul (coada lunga, 76% din operatiile distincte dar doar ~10% din volum) ramane pe editorul uman.
|
||||
US-003 implementeaza exact acest default; `--all` ramane disponibil dar NU e calea aprobata pentru v1.
|
||||
- **D2 — Verificare cloud pe esantionul incert: NU in acest PRD.** Toate sursele sunt suggestion-only (blast
|
||||
radius mic: o sugestie gresita = omul alege altceva in editor). Pre-filtrul NUL (US-001) acopera punctul slab
|
||||
masurat. Codurile rare/avarii grave sunt volum mic; un pas de verificare cloud adauga un backend in plus pentru
|
||||
castig marginal. Se reia DOAR daca esantionul uman (Decision #19, vezi Riscuri) arata ca erorile pe coduri rare
|
||||
sunt o problema reala. `source`/`confidence` din seed raman in DB pentru o eventuala flag-uire ulterioara.
|
||||
- **D3 — Pastram exact-match (SILVER) separat de k-NN.** Exact-match (`lookup_suggestion` pe text normalizat) =
|
||||
instant, 100% pe text identic; k-NN = generalizare semantica pentru texte nevazute. Precedenta confirmata:
|
||||
**GOLD partajat > exact (SILVER) > k-NN embedding** (US-006). k-NN NU inlocuieste exact-match.
|
||||
- **D4 — Bootstrap-ul v1 ruleaza pe LM Studio local** (Qwen3-4B, `json_schema` strict), nu pe Groq/OpenRouter.
|
||||
Motiv: zero cost per-token, date pe hardware propriu (PII service local), masurat 91% pe batch greu + 20/20 validare.
|
||||
Groq/OpenRouter raman in tool ca fallback interschimbabil (US-002), dar nu sunt calea aprobata pentru v1. Cerinta user, 2026-06-28.
|
||||
- **D5 — Dedup pe `normalize_for_match` INAINTE de orice apel LLM, cu reuse in spatiu normalizat.** Nu se trimite la LLM
|
||||
niciun duplicat normalizat si nicio operatie deja etichetata. Garantat de `test_zero_duplicate_trimis_la_llm` (within-run) +
|
||||
`test_rerun_zero_apeluri_llm` (cross-run, idempotenta) — US-003.
|
||||
Motiv: ~31% randuri redundante (19.456 brute -> 13.519 de etichetat: cross-file + variatii spatii + reuse labels existente);
|
||||
fara dedup-ul corect platim apeluri LLM inutile si riscam etichete inconsistente pe acelasi text logic. Cerinta user, 2026-06-28.
|
||||
|
||||
## 6. Valuri de executie (graful de dependente)
|
||||
|
||||
```
|
||||
PASUL 1 — BOOTSTRAP ETICHETE OFFLINE (LM Studio LLM) — fundatia, ruleaza prima:
|
||||
Val 1: [US-002] [US-001] ← US-002 (etichetator LM Studio) = pasul 1; US-001 (pre-filtru NUL) paralel, fisiere disjuncte
|
||||
Val 2: [US-003] ← deblocat de US-002: dedup normalizat -> trimite la LLM -> seed comis
|
||||
PASUL 2 — CONSUM SEED (fara LLM):
|
||||
Val 3: [US-004] ← deblocat de US-003 (owns schema/seed loader)
|
||||
Val 4: [US-005] ← deblocat de US-004
|
||||
Val 5: [US-006] ← deblocat de US-001 + US-005
|
||||
Val 6: [US-007] (optional) ← deblocat de US-006
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Raport VERIFY (2026-06-29) — PASS
|
||||
|
||||
> Faza VERIFY + CLOSE rulata pe `feat/5.18-corpus-knn-exemple-etichetate`, commit-uri
|
||||
> `756f777` (5.18 core + seed) + `308fee6` (fix lateral start-test ONNX). Seed-ul real produs
|
||||
> cu subagenti Haiku (decizie user 2026-06-29), NU LM Studio (GPU jos) si NU Groq — vezi
|
||||
> nota la "Seed real" mai jos. Abaterea de la D4 (LM Studio = backend bootstrap v1) e
|
||||
> documentata si justificata: motorul de etichetare s-a schimbat, garantiile de calitate
|
||||
> (validare 157 op Haiku vs Groq) sunt mai bune, restul pipeline-ului (US-003..006) e neatins.
|
||||
|
||||
### PASS/FAIL per story
|
||||
|
||||
| Story | Stare | Dovada |
|
||||
|-------|-------|--------|
|
||||
| US-001 pre-filtru NUL | PASS | `tests/test_prefiltru_nul.py` verde; seed contine 2200 NUL (`is_nul=1`, `cod=NULL`) |
|
||||
| US-002 etichetator offline | PASS | `tests/test_eticheteaza_tool.py` verde (json_schema envelope, enum cod, scrub PII, no_think) |
|
||||
| US-003 generare seed pe frecventa | PASS | `tests/test_genereaza_seed.py` verde (dedup normalizat, zero-duplicat, idempotenta cross-run, conflict determinist) |
|
||||
| US-004 seeder DB | PASS | `tests/test_operatii_seed.py` verde; smoke `init_db` pe DB gol -> `mapping_suggestions`=17181, NUL=2200, re-seed = 0 inserate (idempotent) |
|
||||
| US-005 embeddings pe corpus etichetat | PASS | `tests/test_embeddings_corpus_etichetat.py` verde (corpus din `mapping_suggestions`, query normalizat simetric, `is_nul` propagat) |
|
||||
| US-006 enrich = NUL + exact + k-NN | PASS | `tests/test_enrich_corpus_etichetat.py` verde (precedenta NUL>GOLD>exact>k-NN, abtinere sub prag, invariant #13 regresie) |
|
||||
| US-007 badge sursa (optional) | PASS | `tests/test_web_badge_sursa.py` verde (4 teste); E2E render live confirma chip confirmat/similar/non-operatie. Implementat la cererea user (2026-06-29) |
|
||||
|
||||
### Dovezi agregat
|
||||
|
||||
- **Suita completa**: `python3 -m pytest -q -m "not live"` -> **1387 passed, 1 deselected (live), 0 failed** (142.77s).
|
||||
- **Cele 6 fisiere de test 5.18** rulate izolat: **36 passed** (`test_prefiltru_nul`, `test_eticheteaza_tool`, `test_genereaza_seed`, `test_operatii_seed`, `test_embeddings_corpus_etichetat`, `test_enrich_corpus_etichetat`).
|
||||
- **Smoke seeder** (`init_db` pe DB gol, `AUTOPASS_SEED_OPERATII_ENABLED=true`): 17181 randuri in `mapping_suggestions`, 2200 NUL, `source='haiku_seed'`, re-seed idempotent (0 inserate).
|
||||
- **Validare nomenclator**: toate codurile distincte din seed (`OE-1`..`OE-8`, `OE-I/R`, `AITLV`, `R-ODO`) sunt in `FALLBACK_NOMENCLATOR` — zero cod gunoi care ar da HTTP 500 / `ORA-12899` la RAR.
|
||||
|
||||
### Seed real (abatere de la D4, aprobata de user)
|
||||
|
||||
Seed-ul `app/data/operatii-etichetate.json` rescris de la 3758 (Groq partial) la **17181** operatii
|
||||
distincte (toate, ordine frecventa), `source="haiku_seed"`, prin subagenti Haiku in Claude Code
|
||||
(blocantul GPU LM Studio rezolvat fara GPU). Validare la dezacorduri Haiku vs Groq pe 157 operatii:
|
||||
Haiku corect ~22/30, Groq ~0 (ex: CHIRIE ANVELOPE->NUL, ADAPTARE electronica->OE-7, INLOCUIT
|
||||
PLACUTE FRANA->OE-1). Distributie: OE-1=13764 (cap, asteptat), NUL=2200, restul sparse. Calitate
|
||||
estimata la scara ~95%; codurile rare (avarii grave OE-C/S/D/F/A, OE-5/6) sunt sparse si pot avea
|
||||
erori de margine ne-verificate uman — ramane recomandarea Decision #19 (esantion uman) inainte de
|
||||
orice crestere de incredere / auto-send.
|
||||
|
||||
### CLOSE — `/code-review high` (main..HEAD, 3 finder x 8 unghiuri)
|
||||
|
||||
Calea de runtime in productie = **curata**. Verificat intact:
|
||||
- **Invariant #13**: nimic din SILVER/k-NN/NUL nu intra in `resolve_prestatii`/`load_mapping` (suggestion-only).
|
||||
- `suggest_nearest`/`enrich_suggestions` semnatura noua (`is_nul`) consumata corect de unicul apelant.
|
||||
- Worker keepalive RAR (`308fee6`/`c05fa00`): fara race (worker single-thread), heartbeat actualizat doar pe login reusit.
|
||||
- Config `embeddings_enabled=True` + `seed_operatii_enabled=True` default: teste neafectate (conftest override).
|
||||
|
||||
Findings (toate low / cosmetic, niciun bug de runtime) — **REPARATE in faza CLOSE**:
|
||||
1. `tools/mapare-llm/genereaza_seed.py` (`_incarca_seed`/`construieste_harta_etichete`): `json.loads(open(...).read())` fara context manager -> FD leak in tool offline. **Fix**: `with open(...)`.
|
||||
2. `app/shared_store.py` `seed_suggestions`: `cod=" "` (whitespace) -> `''` in loc de NULL pe rand non-NUL. **Fix**: `str(...).strip().upper() or None` INAINTE de truthiness. Lock: `test_seed_suggestions_cod_whitespace_devine_null`.
|
||||
3. `app/embeddings.py` (2 docstring-uri): ziceau `[{cod, similaritate}]`, real `[{cod, is_nul, similaritate}]`. **Fix**: docstring-uri aliniate.
|
||||
|
||||
Concluzie VERIFY: **PASS**. US-001..006 livrate cu dovezi; zero bug de corectitudine in runtime; cele 3 findings de cleanup reparate + lock-uite.
|
||||
|
||||
### CLOSE — US-007 implementat (cerere user 2026-06-29)
|
||||
|
||||
User a cerut la poarta CLOSE sa includem badge-ul direct pe sugestiile sistemului fuzzy.
|
||||
Implementat: chip in coloana "Sugestii" din `_mapari.html`, mapat din `sugestie_principala.sursa`:
|
||||
**confirmat** (GOLD partajat) / **similar** (SILVER exact + k-NN embeddings) / **non-operatie**
|
||||
(pre-filtru NUL / vecin NUL). CSS `.sugg-sursa--{confirmat,similar,nul}` pe tokeni de tema
|
||||
(`--ok`/`--accent`/`--muted` cu `color-mix`), nu rupe layoutul. Suggestion-only (#13). Fix lateral:
|
||||
`surse_sugestie` default in `routes.py` a primit cheia `nul` (lipsea — finding cross-file). Teste:
|
||||
`tests/test_web_badge_sursa.py` (gold/silver/nul/fara-sursa). Render verificat in serverul real
|
||||
(`/_fragments/mapari`): OP-REV->confirmat, OP-REP->similar, OP-ITP->non-operatie, OP-XYZ->fara chip.
|
||||
Suita: **1392 passed, 1 deselected (live)**.
|
||||
|
||||
---
|
||||
|
||||
<!-- AUTONOMOUS DECISION LOG -->
|
||||
## GSTACK REVIEW REPORT (/autoplan — Eng focus, 2026-06-28)
|
||||
|
||||
Scope review: Eng (CEO premise gate + Eng dual-voice). Design/DX sarite (UI = doar badge optional US-007, tool intern mono-dezvoltator). Voce Eng: **subagent-only** — Codex a lovit limita de utilizare (degradare conform matricei).
|
||||
|
||||
**Premise confirmate** (poarta umana): (1) k-NN peste exemple reale > 18 categorii generice (94.3% vs 86.2% masurat); (2) etichetare LLM o singura data, offline, zero LLM la runtime; (3) SILVER populat in productie din seed comis; (4) pre-filtru NUL necesar (recall 64%); (5) LM Studio Qwen3-4B = calitate acceptabila pt bootstrap (91% batch greu / 20/20 validare).
|
||||
|
||||
**Cerinte user incorporate**: D4 (LM Studio = backend default v1), D5 (dedup pe `normalize_for_match` + reuse normalizat, INAINTE de LLM).
|
||||
|
||||
### Decision Audit Trail
|
||||
|
||||
| # | Faza | Decizie | Clasif. | Principiu | Rationament |
|
||||
|---|------|---------|---------|-----------|-------------|
|
||||
| 1 | CEO | Restructurare valuri: Pasul 1 = bootstrap LM Studio (US-002->US-003) | Mecanic | P1 | Cerinta user explicita; reflecta dependenta reala |
|
||||
| 2 | Eng | F1: query embedding normalizat ca si corpusul (US-005 AC + test) | Mecanic | P5 | Corectitudine; altfel 94.3% nereproductibil. Blast radius (US-005) |
|
||||
| 3 | Eng | F2: seed comis = cache de etichete cross-run (US-003 pipeline + `test_rerun_zero_apeluri_llm`) | Mecanic | P1 | Criteriul "0 apel LLM la re-run" altfel nesatisfiabil |
|
||||
| 4 | Eng | F3: harta normalizat->cod cu tie-break determinist (freq-max) | Mecanic | P5 | 1 conflict real azi (CURATAT CATALIZATOR); altfel cod nedeterminist |
|
||||
| 5 | Eng | F4/F5: corectie cifre (17.181 distinct, 13.519 de etichetat, 31%) + claim "fara punctuatie" | Mecanic | P5 | Cifre verificate cu `normalize_for_match` real |
|
||||
| 6 | Eng | F6: arunca cheie normalizata vida inainte de dedup | Mecanic | P1 | Coliziune pe slot UNIQUE gol |
|
||||
| 7 | Eng | F7: teste two-run + conflict adaugate | Mecanic | P1 | Testul single-run nu acopera idempotenta/determinismul |
|
||||
| 8 | Eng | F8: envelope json_schema strict + enum cod + dezactivare thinking Qwen3 + garda truncare | Mecanic | P1 | Realism integrare LM Studio (cerinta user #1) |
|
||||
| 9 | Eng | F9: parsare NR toleranta (skip, nu zero-weight) | Mecanic | P3 | Date curate azi; ieftina robustete |
|
||||
| 10 | Eng | F10: re-justificare INSERT OR IGNORE (confirmari umane = shared_mappings) | Mecanic | P5 | Evita inducerea in eroare a unui mentainer |
|
||||
|
||||
Zero decizii de gust (taste) si zero user-challenge: toate constatarile au intarit directia user, nu au contrazis-o.
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user