align for IA agents + grafana

This commit is contained in:
Nicolas Cantu 2025-09-22 14:02:07 +00:00
parent 3367c53963
commit 3df6f5d7f0
3 changed files with 54 additions and 21 deletions

View File

@ -2,6 +2,7 @@
DOMAIN=dev4.4nkweb.com DOMAIN=dev4.4nkweb.com
BOOTSTRAP_DOMAIN=dev3.4nkweb.com BOOTSTRAP_DOMAIN=dev3.4nkweb.com
LOCAL_DOMAIN=local.4nkweb.com LOCAL_DOMAIN=local.4nkweb.com
LECOFFRE_BACK_DOMAIN=dev3.4nkweb.com
# GIT # GIT
GITEA_BASE_URL=git.4nkweb.com GITEA_BASE_URL=git.4nkweb.com
@ -31,7 +32,7 @@ NEXT_PUBLIC_FRONT_APP_HOST=https://dev4.4nkweb.com/lecoffre
NEXT_PUBLIC_IDNOT_BASE_URL=https://qual-connexion.idnot.fr NEXT_PUBLIC_IDNOT_BASE_URL=https://qual-connexion.idnot.fr
NEXT_PUBLIC_IDNOT_AUTHORIZE_ENDPOINT=/IdPOAuth2/authorize/idnot_idp_v1 NEXT_PUBLIC_IDNOT_AUTHORIZE_ENDPOINT=/IdPOAuth2/authorize/idnot_idp_v1
NEXT_PUBLIC_BACK_API_PROTOCOL=https NEXT_PUBLIC_BACK_API_PROTOCOL=https
NEXT_PUBLIC_BACK_API_HOST=${DOMAIN} NEXT_PUBLIC_BACK_API_HOST=${LECOFFRE_BACK_DOMAIN}
NEXT_PUBLIC_BACK_API_PORT=443 NEXT_PUBLIC_BACK_API_PORT=443
NEXT_PUBLIC_BACK_API_ROOT_URL=/api NEXT_PUBLIC_BACK_API_ROOT_URL=/api
NEXT_PUBLIC_BACK_API_VERSION=v1 NEXT_PUBLIC_BACK_API_VERSION=v1
@ -99,13 +100,19 @@ MAILCHIMP_KEY=3fa54304bc766dfd0b8043a827b28a3a-us17
MAILCHIMP_LIST_ID=a48d9ad852 MAILCHIMP_LIST_ID=a48d9ad852
# Configuration Stripe # Configuration Stripe
STRIPE_SECRET_KEY=sk_test_51OwKmMP5xh1u9BqSeFpqw0Yr15hHtFsh0pvRGaE0VERhlYtvw33ND1qiGA6Dy1DPmmV61B6BqIimlhuv7bwElhjF00PLQwD60n STRIPE_SECRET_KEY=sk_test_51SA9bu0NPxjC8mAm5xdqVR5iKjzXw2QfNXfjYC2KbIJtCfENGj1uLvVCWJ2H7QCnnmLUqqgH45Qh74Sbss9GqDZv00WF9Me4Zp
STRIPE_PUBLISHABLE_KEY=pk_test_51OwKmMP5xh1u9BqSOCBC1H40ACnGE4qH5ZERZ4tKktQuTcqRTPY26QqKeM6FfghdnbcC7nPbtMtnB6jemInrs67Z008RxkUGcn # STRIPE_SECRET_KEY=sk_test_51OwKmMP5xh1u9BqSeFpqw0Yr15hHtFsh0pvRGaE0VERhlYtvw33ND1qiGA6Dy1DPmmV61B6BqIimlhuv7bwElhjF00PLQwD60n
# STRIPE_PUBLISHABLE_KEY=pk_test_51OwKmMP5xh1u9BqSOCBC1H40ACnGE4qH5ZERZ4tKktQuTcqRTPY26QqKeM6FfghdnbcC7nPbtMtnB6jemInrs67Z008RxkUGcn
STRIPE_PUBLISHABLE_KEY=pk_test_51SA9bu0NPxjC8mAm2a6Z01BqdpoLRGup8va1wPnYP0OLhrV6s5ch9rwE0y82oQpCiBJrDPfgab6cl0g4Rh1MPQ9h00f7TJXNhU
STRIPE_WEBHOOK_SECRET= STRIPE_WEBHOOK_SECRET=
STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=price_1P66fuP5xh1u9BqSHj0O6Uy3 # STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=price_1P66fuP5xh1u9BqSHj0O6Uy3
STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NsRP5xh1u9BqSFgkUDbQY STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=price_1SA9qP0NPxjC8mAml8YAm5mQ
STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=price_1P66RqP5xh1u9BqSuUzkQNac # STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NsRP5xh1u9BqSFgkUDbQY
STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NpKP5xh1u9BqSApFogvUB STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1SA9q60NPxjC8mAmvzkoWdnE
# STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=price_1P66RqP5xh1u9BqSuUzkQNac
STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=price_1SA9nI0NPxjC8mAmfQBlQQFl
# STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NpKP5xh1u9BqSApFogvUB
STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1SA9n20NPxjC8mAmZFn2fw9l
price_1QMUuXP5xh1u9BqS26uzbJaF_name='créateurs' price_1QMUuXP5xh1u9BqS26uzbJaF_name='créateurs'
price_1P9O6BP5xh1u9BqSelP9ZI52_name='standard annuel' price_1P9O6BP5xh1u9BqSelP9ZI52_name='standard annuel'
price_1P9O68P5xh1u9BqSfNVdM8QL_name='starter annuel' price_1P9O68P5xh1u9BqSfNVdM8QL_name='starter annuel'

View File

@ -2,6 +2,7 @@
DOMAIN=dev4.4nkweb.com DOMAIN=dev4.4nkweb.com
BOOTSTRAP_DOMAIN=dev3.4nkweb.com BOOTSTRAP_DOMAIN=dev3.4nkweb.com
LOCAL_DOMAIN=local.4nkweb.com LOCAL_DOMAIN=local.4nkweb.com
LECOFFRE_BACK_DOMAIN=dev3.4nkweb.com
# GIT # GIT
GITEA_BASE_URL=git.4nkweb.com GITEA_BASE_URL=git.4nkweb.com
@ -10,8 +11,9 @@ GITEA_OWNER="nicolas.cantu,Omar"
GITEA_RUNNER_NAME=debian-runner GITEA_RUNNER_NAME=debian-runner
# Variables d'environnement pour l'application back-end # Variables d'environnement pour l'application back-end
NODE_OPTIONS=--max-old-space-size=2048
NODE_ENV=production NODE_ENV=production
RUST_LOG=DEBUG
NODE_OPTIONS=--max-old-space-size=2048
# Configuration IDNOT # Configuration IDNOT
IDNOT_ANNUARY_BASE_URL=https://qual-api.notaires.fr/annuaire IDNOT_ANNUARY_BASE_URL=https://qual-api.notaires.fr/annuaire
@ -30,7 +32,7 @@ NEXT_PUBLIC_FRONT_APP_HOST=https://dev4.4nkweb.com/lecoffre
NEXT_PUBLIC_IDNOT_BASE_URL=https://qual-connexion.idnot.fr NEXT_PUBLIC_IDNOT_BASE_URL=https://qual-connexion.idnot.fr
NEXT_PUBLIC_IDNOT_AUTHORIZE_ENDPOINT=/IdPOAuth2/authorize/idnot_idp_v1 NEXT_PUBLIC_IDNOT_AUTHORIZE_ENDPOINT=/IdPOAuth2/authorize/idnot_idp_v1
NEXT_PUBLIC_BACK_API_PROTOCOL=https NEXT_PUBLIC_BACK_API_PROTOCOL=https
NEXT_PUBLIC_BACK_API_HOST=${DOMAIN} NEXT_PUBLIC_BACK_API_HOST=${LECOFFRE_BACK_DOMAIN}
NEXT_PUBLIC_BACK_API_PORT=443 NEXT_PUBLIC_BACK_API_PORT=443
NEXT_PUBLIC_BACK_API_ROOT_URL=/api NEXT_PUBLIC_BACK_API_ROOT_URL=/api
NEXT_PUBLIC_BACK_API_VERSION=v1 NEXT_PUBLIC_BACK_API_VERSION=v1
@ -71,9 +73,6 @@ bitcoin_data_dir=/home/bitcoin/.bitcoin
bootstrap_url=wss://${BOOTSTRAP_DOMAIN}/ws/ bootstrap_url=wss://${BOOTSTRAP_DOMAIN}/ws/
bootstrap_faucet=true bootstrap_faucet=true
RUST_LOG=DEBUG,
NODE_OPTIONS=--max-old-space-size=2048
# ================== /!\ sensible ========================= # ================== /!\ sensible =========================
# Configuration IDNOT # Configuration IDNOT
@ -101,13 +100,19 @@ MAILCHIMP_KEY=3fa54304bc766dfd0b8043a827b28a3a-us17
MAILCHIMP_LIST_ID=a48d9ad852 MAILCHIMP_LIST_ID=a48d9ad852
# Configuration Stripe # Configuration Stripe
STRIPE_SECRET_KEY=sk_test_51OwKmMP5xh1u9BqSeFpqw0Yr15hHtFsh0pvRGaE0VERhlYtvw33ND1qiGA6Dy1DPmmV61B6BqIimlhuv7bwElhjF00PLQwD60n STRIPE_SECRET_KEY=sk_test_51SA9bu0NPxjC8mAm5xdqVR5iKjzXw2QfNXfjYC2KbIJtCfENGj1uLvVCWJ2H7QCnnmLUqqgH45Qh74Sbss9GqDZv00WF9Me4Zp
STRIPE_PUBLISHABLE_KEY= # STRIPE_SECRET_KEY=sk_test_51OwKmMP5xh1u9BqSeFpqw0Yr15hHtFsh0pvRGaE0VERhlYtvw33ND1qiGA6Dy1DPmmV61B6BqIimlhuv7bwElhjF00PLQwD60n
# STRIPE_PUBLISHABLE_KEY=pk_test_51OwKmMP5xh1u9BqSOCBC1H40ACnGE4qH5ZERZ4tKktQuTcqRTPY26QqKeM6FfghdnbcC7nPbtMtnB6jemInrs67Z008RxkUGcn
STRIPE_PUBLISHABLE_KEY=pk_test_51SA9bu0NPxjC8mAm2a6Z01BqdpoLRGup8va1wPnYP0OLhrV6s5ch9rwE0y82oQpCiBJrDPfgab6cl0g4Rh1MPQ9h00f7TJXNhU
STRIPE_WEBHOOK_SECRET= STRIPE_WEBHOOK_SECRET=
STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=price_1P66fuP5xh1u9BqSHj0O6Uy3 # STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=price_1P66fuP5xh1u9BqSHj0O6Uy3
STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NsRP5xh1u9BqSFgkUDbQY STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=price_1SA9qP0NPxjC8mAml8YAm5mQ
STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=price_1P66RqP5xh1u9BqSuUzkQNac # STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NsRP5xh1u9BqSFgkUDbQY
STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NpKP5xh1u9BqSApFogvUB STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1SA9q60NPxjC8mAmvzkoWdnE
# STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=price_1P66RqP5xh1u9BqSuUzkQNac
STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=price_1SA9nI0NPxjC8mAmfQBlQQFl
# STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1P9NpKP5xh1u9BqSApFogvUB
STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=price_1SA9n20NPxjC8mAmZFn2fw9l
price_1QMUuXP5xh1u9BqS26uzbJaF_name='créateurs' price_1QMUuXP5xh1u9BqS26uzbJaF_name='créateurs'
price_1P9O6BP5xh1u9BqSelP9ZI52_name='standard annuel' price_1P9O6BP5xh1u9BqSelP9ZI52_name='standard annuel'
price_1P9O68P5xh1u9BqSfNVdM8QL_name='starter annuel' price_1P9O68P5xh1u9BqSfNVdM8QL_name='starter annuel'

View File

@ -211,6 +211,12 @@ def get_storage_size_bytes(container: str) -> int:
return {} return {}
def docker_ps_names() -> set:
code, out, _ = run_cmd(["docker", "ps", "--format", "{{.Names}}"], timeout_seconds=6)
if code != 0 or not out:
return set()
return set(n.strip() for n in out.splitlines() if n.strip())
def ovh_safe_check(app_key: str, app_secret: str, consumer_key: str, service_name: str, base_url: str = "https://eu.api.ovh.com/1.0") -> dict: def ovh_safe_check(app_key: str, app_secret: str, consumer_key: str, service_name: str, base_url: str = "https://eu.api.ovh.com/1.0") -> dict:
try: try:
# Get OVH time # Get OVH time
@ -401,10 +407,20 @@ class StatusAPIHandler(BaseHTTPRequestHandler):
except Exception: except Exception:
pass pass
# Deployment progress: percentage of services healthy/running # Deployment progress: basé sur présence (docker ps) ET santé
running_names = docker_ps_names()
total = len(services) total = len(services)
healthy = sum(1 for s in services if s.get("health") == "healthy" or s.get("status") == "running") ready = 0
percent = int(healthy * 100 / total) if total else 0 for s in services:
name = s.get("name")
container = next((d["container"] for d in service_defs if d["name"] == name), None)
present = container in running_names if container else False
healthy_s = (s.get("health") == "healthy")
running_s = (s.get("status") == "running")
# considérer prêt si present ET (healthy ou running)
if present and (healthy_s or running_s):
ready += 1
percent = int(ready * 100 / total) if total else 0
# Integrations: Mailchimp (Mandrill ping) and Stripe (counts) # Integrations: Mailchimp (Mandrill ping) and Stripe (counts)
mailchimp_test = {"provider": "Mailchimp", "status": "missing"} mailchimp_test = {"provider": "Mailchimp", "status": "missing"}
@ -426,6 +442,7 @@ class StatusAPIHandler(BaseHTTPRequestHandler):
# Stripe: lister prices et agréger en balayant les subscriptions (sans filtre price) # Stripe: lister prices et agréger en balayant les subscriptions (sans filtre price)
stripe_by_offer = {"CREATORS": 0, "STARTER": 0, "STANDARD": 0, "UNLIMITED": 0, "TOTAL": 0} stripe_by_offer = {"CREATORS": 0, "STARTER": 0, "STANDARD": 0, "UNLIMITED": 0, "TOTAL": 0}
stripe_prices_map = {} stripe_prices_map = {}
stripe_price_counts = {}
if env_map.get("STRIPE_SECRET_KEY"): if env_map.get("STRIPE_SECRET_KEY"):
try: try:
auth_h = f"Authorization: Bearer {env_map.get('STRIPE_SECRET_KEY')}" auth_h = f"Authorization: Bearer {env_map.get('STRIPE_SECRET_KEY')}"
@ -439,6 +456,7 @@ class StatusAPIHandler(BaseHTTPRequestHandler):
for pr in prices: for pr in prices:
pid = pr.get('id') pid = pr.get('id')
stripe_prices_map[pid] = pr.get('nickname') or '' stripe_prices_map[pid] = pr.get('nickname') or ''
stripe_price_counts[pid] = 0
# Déterminer les familles par ID connus (si présents dans l'env) sinon par nickname # Déterminer les familles par ID connus (si présents dans l'env) sinon par nickname
creators_ids = set(filter(None, [env_map.get("STRIPE_CREATORS_PRICE_ID")])) creators_ids = set(filter(None, [env_map.get("STRIPE_CREATORS_PRICE_ID")]))
standard_ids = set(filter(None, [ standard_ids = set(filter(None, [
@ -489,6 +507,8 @@ class StatusAPIHandler(BaseHTTPRequestHandler):
continue continue
stripe_by_offer[fam] = stripe_by_offer.get(fam, 0) + 1 stripe_by_offer[fam] = stripe_by_offer.get(fam, 0) + 1
stripe_by_offer["TOTAL"] += 1 stripe_by_offer["TOTAL"] += 1
if pid:
stripe_price_counts[pid] = stripe_price_counts.get(pid, 0) + 1
if d.get("has_more") and subs: if d.get("has_more") and subs:
starting_after = subs[-1].get('id') starting_after = subs[-1].get('id')
pages += 1 pages += 1
@ -572,6 +592,7 @@ class StatusAPIHandler(BaseHTTPRequestHandler):
"ovh": ovh_test, "ovh": ovh_test,
"mailchimp": mailchimp_test, "mailchimp": mailchimp_test,
"stripe_subscriptions_by_offer": stripe_by_offer, "stripe_subscriptions_by_offer": stripe_by_offer,
"stripe_prices": {pid: {"nickname": stripe_prices_map.get(pid, ""), "count": cnt} for pid, cnt in stripe_price_counts.items()},
}, },
"wallets": wallets, "wallets": wallets,
} }