feat(account): CLI lifecycle conturi + accounts.active (PRD 3.1)
Inlocuieste crearea conturilor prin INSERT SQL manual cu un tool admin dedicat, simetric cu tools/apikey.py. Fundatia Etapei 3 (3.2/3.3). - app/accounts.py: create_account/set_active/list_accounts (helper pur, partajat CLI + viitor flux web 3.3). Normalizeaza CUI (trim+upper), prinde IntegrityError -> ValueError cu cauza+fix. - accounts.active (lifecycle cont) + index unic partial ux_accounts_cui (unicitate la nivel de index, fara fereastra de coliziune). Migrare idempotenta in _migrate. - tools/account.py: create (--name/--cui/--inactive/--with-key atomic), list [--pending], activate/deactivate --account N. Erori -> exit 2. - 20 teste noi (12 helper + 8 CLI); suita 299 passed. active e inert pana la gate-ul worker din 3.3 (documentat). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
124
tools/account.py
Normal file
124
tools/account.py
Normal file
@@ -0,0 +1,124 @@
|
||||
#!/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
|
||||
from app.auth import create_api_key
|
||||
from app.db import get_connection, init_db
|
||||
|
||||
|
||||
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, args.cui, 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, args.cui, 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 _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", default=None, help="CUI (unic cand e prezent)")
|
||||
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")
|
||||
|
||||
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)
|
||||
finally:
|
||||
conn.close()
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
sys.exit(main())
|
||||
Reference in New Issue
Block a user