PRD 5.16 — propagare design finalizata (system font stack, fara IBM Plex self-hostat): - US-001/002/008: tokeni --font-ui/--font-mono (system stack) + scala --fs-*; zero @font-face si zero /static/fonts/; landing aliniat la acelasi stack - US-003: RAR online = dot compact in antet + meniu burger; banda rosie DOAR pe blocat (invariant zero-silent-failures pastrat) - US-010: antet "ROMFAST AUTOPASS" + nume service + /login brandeit 2 coloane + badge plan; meniu burger cu separatoare; gate strict pe is_authenticated - US-011: selector tema pill icon+eticheta (reuse THEMES) - US-004/005/006/007: bug-fix editor prestatii (picker cod+denumire, add_extra in mod operatii, cod ales se salveaza fara "+", Renunta inchide via closest) - US-012/013: landing Autentificare->/login; wizard import colapsat + 4 pasi pe tokeni - fix VERIFY E2E: contoare duplicate pe 390px (inline display:flex batea @media) -> CSS + test-lock PRD 5.17 — tipuri de cont + trial Pro 30z + enforcement DUR: - US-001/002/008: accounts.tier + trial_until (migrare aditiva defensiva); app/plans.py sursa unica (PLANS, FREE_MONTHLY_LIMIT=60, effective_tier(now injectabil), monthly_usage, CONSUMED_STATUSES); create_account trial Pro 30z; CLI set-tier (protejat id=1, audit) - US-003/004/005: enforce volum 60/luna INAINTE de build_key pe ambele canale (PLAN_LIMITA_LUNARA, 3 niveluri + log_event); gate API Pro+ (PLAN_FARA_API 403 actionabil); valideaza/nomenclator raman permise; downgrade lazy; flag AUTOPASS_ENFORCE_PLANS (kill-switch) - US-006: badge plan antet + linie burger + consum N/60 + warn>=80% + 6 stari + copy RO pluralizat + banner one-time trial->Gratuit + pagina Cont Regresie: 1380 passed, 0 failed, 1 deselected (live). E2E browser pe 390/1280 confirmat. Backend trimitere (worker/masina stari/idempotenta/contract RAR) NEATINS. Lucrul 5.18 (corpus kNN) ramane separat, necomis. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
188 lines
7.2 KiB
Python
188 lines
7.2 KiB
Python
#!/usr/bin/env python3
|
|
"""CLI lifecycle conturi ROAAUTO (admin gateway).
|
|
|
|
Onboardeaza/activeaza un client fara INSERT SQL manual, simetric cu
|
|
`tools/apikey.py`. Adminul ruleaza pe masina gateway — nicio suprafata HTTP de
|
|
admin (admin web vine in 3.3). Optional emite si prima cheie API intr-un pas
|
|
(`--with-key`), atomic cu crearea contului.
|
|
|
|
NOTA: `deactivate` comuta `accounts.active` (lifecycle), dar NU opreste inca
|
|
trimiterile — gate-ul worker pe `active` apartine 3.3. Vezi `app/accounts.py`.
|
|
|
|
Utilizare:
|
|
python -m tools.account create --name "Service X" [--cui RO123] [--inactive] [--with-key]
|
|
python -m tools.account list [--pending]
|
|
python -m tools.account activate --account 2
|
|
python -m tools.account deactivate --account 2
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import argparse
|
|
import sqlite3
|
|
import sys
|
|
|
|
from app.accounts import create_account, list_accounts, set_active, set_tier
|
|
from app.auth import create_api_key
|
|
from app.db import get_connection, init_db
|
|
from app.users import set_admin
|
|
|
|
|
|
def _create(conn: sqlite3.Connection, args: argparse.Namespace) -> int:
|
|
active = not args.inactive
|
|
if not args.with_key:
|
|
try:
|
|
acct_id = create_account(conn, args.name, cui=args.cui, email=args.email, active=active)
|
|
except ValueError as exc:
|
|
print(f"eroare: {exc}", file=sys.stderr)
|
|
return 2
|
|
print(f"Cont creat: id={acct_id} (activ={'da' if active else 'nu'})")
|
|
return 0
|
|
|
|
# --with-key: cont + cheie in aceeasi tranzactie (DB ruleaza autocommit).
|
|
conn.execute("BEGIN IMMEDIATE")
|
|
try:
|
|
acct_id = create_account(conn, args.name, cui=args.cui, email=args.email, active=active)
|
|
key = create_api_key(conn, acct_id)
|
|
conn.execute("COMMIT")
|
|
except ValueError as exc:
|
|
conn.execute("ROLLBACK")
|
|
print(f"eroare: {exc}", file=sys.stderr)
|
|
return 2
|
|
except Exception:
|
|
conn.execute("ROLLBACK")
|
|
raise
|
|
print(f"Cont creat: id={acct_id} (activ={'da' if active else 'nu'})")
|
|
print("Cheie API (pastreaz-o, nu se mai afiseaza):")
|
|
print(key)
|
|
return 0
|
|
|
|
|
|
def _set_active(conn: sqlite3.Connection, account_id: int, active: bool) -> int:
|
|
try:
|
|
set_active(conn, account_id, active)
|
|
except ValueError as exc:
|
|
print(f"eroare: {exc}", file=sys.stderr)
|
|
return 2
|
|
print(f"Cont {account_id}: activ={'da' if active else 'nu'}")
|
|
return 0
|
|
|
|
|
|
def _set_tier(conn: sqlite3.Connection, account_id: int, tier: str, trial_until: str | None) -> int:
|
|
try:
|
|
set_tier(conn, account_id, tier, trial_until=trial_until)
|
|
except ValueError as exc:
|
|
print(f"eroare: {exc}", file=sys.stderr)
|
|
return 2
|
|
trial_msg = f", trial_until={trial_until}" if trial_until else ", fara trial"
|
|
print(f"Cont {account_id}: tier={tier}{trial_msg}")
|
|
return 0
|
|
|
|
|
|
def _set_admin(conn: sqlite3.Connection, account_id: int, is_admin: bool) -> int:
|
|
try:
|
|
set_admin(conn, account_id, is_admin=is_admin)
|
|
except ValueError as exc:
|
|
print(f"eroare: {exc}", file=sys.stderr)
|
|
return 2
|
|
actiune = "admin" if is_admin else "non-admin"
|
|
print(f"Cont {account_id}: marcat ca {actiune}")
|
|
return 0
|
|
|
|
|
|
def _list(conn: sqlite3.Connection, pending_only: bool) -> int:
|
|
rows = list_accounts(conn)
|
|
if pending_only:
|
|
rows = [r for r in rows if not r["active"]]
|
|
if not rows:
|
|
print("(niciun cont in asteptare)" if pending_only else "(niciun cont)")
|
|
return 0
|
|
print(f"{'id':>4} {'activ':>5} {'cui':<14} {'creat':<20} nume")
|
|
for r in rows:
|
|
print(
|
|
f"{r['id']:>4} {('da' if r['active'] else 'nu'):>5} "
|
|
f"{(r['cui'] or ''):<14} {(r['created_at'] or ''):<20} {r['name']}"
|
|
)
|
|
return 0
|
|
|
|
|
|
def main(argv: list[str] | None = None) -> int:
|
|
parser = argparse.ArgumentParser(description="Lifecycle conturi gateway RAR AUTOPASS")
|
|
sub = parser.add_subparsers(dest="cmd", required=True)
|
|
|
|
p_create = sub.add_parser("create", help="creeaza un cont nou")
|
|
p_create.add_argument("--name", required=True, help="nume cont (service)")
|
|
p_create.add_argument("--cui", required=True, help="CUI firma (obligatoriu, unic)")
|
|
p_create.add_argument("--email", required=True, help="email de contact al firmei (obligatoriu)")
|
|
p_create.add_argument("--inactive", action="store_true", help="creeaza cont in asteptare (active=0)")
|
|
p_create.add_argument("--with-key", action="store_true", help="emite si prima cheie API (atomic)")
|
|
|
|
p_list = sub.add_parser("list", help="listeaza conturi")
|
|
p_list.add_argument("--pending", action="store_true", help="doar conturi in asteptare (active=0)")
|
|
|
|
p_act = sub.add_parser("activate", help="activeaza un cont")
|
|
p_act.add_argument("--account", type=int, required=True, help="account_id")
|
|
|
|
p_deact = sub.add_parser("deactivate", help="dezactiveaza un cont")
|
|
p_deact.add_argument("--account", type=int, required=True, help="account_id")
|
|
|
|
p_sadmin = sub.add_parser("set-admin", help="seteaza/sterge rol admin pe un cont")
|
|
p_sadmin.add_argument("--account", type=int, required=True, help="account_id")
|
|
p_sadmin.add_argument("--remove", action="store_true", help="sterge rolul admin (implicit: adauga)")
|
|
|
|
p_stier = sub.add_parser(
|
|
"set-tier",
|
|
help="seteaza planul unui cont (free/standard/pro/premium)",
|
|
description=(
|
|
"Aloca manual un plan de cont. Tier invalid -> eroare clara. "
|
|
"Contul de sistem id=1 e protejat."
|
|
),
|
|
)
|
|
p_stier.add_argument("--account", type=int, required=True, help="account_id")
|
|
p_stier.add_argument(
|
|
"--tier", required=True,
|
|
help="planul de alocat: free | standard | pro | premium"
|
|
)
|
|
_trial_grp = p_stier.add_mutually_exclusive_group()
|
|
_trial_grp.add_argument(
|
|
"--trial-days", type=int, metavar="N",
|
|
help="seteaza trial_until = acum + N zile"
|
|
)
|
|
_trial_grp.add_argument(
|
|
"--no-trial", action="store_true",
|
|
help="sterge trial-ul (trial_until=NULL)"
|
|
)
|
|
|
|
args = parser.parse_args(argv)
|
|
|
|
init_db() # asigura schema (accounts.active + index CUI) + cont default
|
|
conn = get_connection()
|
|
try:
|
|
if args.cmd == "create":
|
|
return _create(conn, args)
|
|
if args.cmd == "list":
|
|
return _list(conn, args.pending)
|
|
if args.cmd == "activate":
|
|
return _set_active(conn, args.account, True)
|
|
if args.cmd == "deactivate":
|
|
return _set_active(conn, args.account, False)
|
|
if args.cmd == "set-admin":
|
|
return _set_admin(conn, args.account, is_admin=not args.remove)
|
|
if args.cmd == "set-tier":
|
|
# Calculeaza trial_until din --trial-days sau None daca --no-trial
|
|
from datetime import datetime, timedelta, timezone
|
|
trial_until: str | None = None
|
|
if getattr(args, "trial_days", None):
|
|
trial_until = (
|
|
datetime.now(timezone.utc) + timedelta(days=args.trial_days)
|
|
).strftime("%Y-%m-%d %H:%M:%S")
|
|
# daca nici --trial-days nici --no-trial -> trial_until=None (fara trial)
|
|
return _set_tier(conn, args.account, args.tier, trial_until)
|
|
finally:
|
|
conn.close()
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
sys.exit(main())
|