#!/usr/bin/env python3 """ Retry v2: Single-cell writes with fresh token, proper error handling """ import json, sys, time, re import psycopg2, psycopg2.extras import requests sys.path.insert(0, '/root/.openclaw/workspace/scripts') from phone_encrypt import encrypt_phone SPREADSHEET_TOKEN = "NoZqsFi47hIOHEt9j8WcfRtbnug" SHEET_ID = "qJF4I" FEISHU_APP_ID = "cli_a929ae22e0b8dcc8" FEISHU_APP_SECRET = "OtFjMy7p3qE3VvLbMdcWidwgHOnGD4FJ" PG_HOST = "bj-postgres-16pob4sg.sql.tencentcdb.com" PG_PORT = 28591 PG_USER = "ai_member" PG_PASSWORD = "LdfjdjL83h3h3^$&**YGG*" PG_DB = "vala_bi" def get_token(): r = requests.post('https://open.feishu.cn/open-apis/auth/v3/tenant_access_token/internal', json={"app_id": FEISHU_APP_ID, "app_secret": FEISHU_APP_SECRET}) return r.json()['tenant_access_token'] def write_cell(token, row_num, col_letter, value): """Write a single cell. Returns True on success.""" range_str = f"{SHEET_ID}!{col_letter}{row_num}:{col_letter}{row_num}" body = {"valueRange": {"range": range_str, "values": [[value]]}} for attempt in range(3): try: r = requests.put( f'https://open.feishu.cn/open-apis/sheets/v2/spreadsheets/{SPREADSHEET_TOKEN}/values', headers={'Authorization': f'Bearer {token}', 'Content-Type': 'application/json'}, json=body, timeout=15 ) text = r.text.strip() try: resp = json.loads(text) except json.JSONDecodeError: print(f" JSON parse error for [{range_str}]: {text[:200]}") time.sleep(1) continue if resp.get('code') == 0: return True elif resp.get('code') == 90217: wait = (attempt + 1) * 3 time.sleep(wait) elif resp.get('code') == 99991663: # token expired return 'TOKEN_EXPIRED' else: if attempt < 2: time.sleep(1) else: print(f" FAIL [{range_str}]: {resp.get('code')} {resp.get('msg')}") except Exception as e: if attempt < 2: time.sleep(1) else: print(f" EXCEPTION [{range_str}]: {e}") return False TOKEN = get_token() print(f"Token: {TOKEN[:20]}...") # Read sheet r = requests.get( f'https://open.feishu.cn/open-apis/sheets/v2/spreadsheets/{SPREADSHEET_TOKEN}/values/{SHEET_ID}!A3:V2512?valueRenderOption=ToString', headers={'Authorization': f'Bearer {TOKEN}'} ) rows = r.json()['data']['valueRange']['values'] print(f"Read {len(rows)} rows") # Extract phones phone_map = {} for i, row in enumerate(rows): if len(row) > 4 and row[4]: phone = str(row[4]).strip() if re.match(r'^1\d{10}$', phone): phone_map[i] = phone print(f"Found {len(phone_map)} phones") # Encrypt enc_to_idx = {} for idx, phone in phone_map.items(): try: enc = encrypt_phone(phone) enc_to_idx[enc] = idx except: pass # Match UIDs conn = psycopg2.connect(host=PG_HOST, port=PG_PORT, user=PG_USER, password=PG_PASSWORD, dbname=PG_DB, connect_timeout=30) cur = conn.cursor(cursor_factory=psycopg2.extras.DictCursor) enc_list = list(enc_to_idx.keys()) uid_map = {} batch_size = 500 for start in range(0, len(enc_list), batch_size): batch = enc_list[start:start+batch_size] placeholders = ','.join(['%s'] * len(batch)) cur.execute(f""" SELECT id, tel_encrypt, created_at::date, download_channel FROM bi_vala_app_account WHERE tel_encrypt IN ({placeholders}) AND status = 1 AND deleted_at IS NULL """, batch) for row in cur.fetchall(): enc = row['tel_encrypt'] if enc in enc_to_idx: idx = enc_to_idx[enc] uid_map[idx] = {'uid': row['id'], 'created_at': str(row['created_at']) if row['created_at'] else None, 'download_channel': row['download_channel']} print(f"Matched {len(uid_map)} UIDs") # Trial counts uid_list = list(set(v['uid'] for v in uid_map.values())) trial_counts = {} for start in range(0, len(uid_list), batch_size): batch = uid_list[start:start+batch_size] placeholders = ','.join(['%s'] * len(batch)) cur.execute(f""" SELECT account_id, COUNT(*) as cnt FROM bi_user_course_detail WHERE account_id IN ({placeholders}) AND expire_time IS NULL AND deleted_at IS NULL GROUP BY account_id """, batch) for row in cur.fetchall(): trial_counts[row['account_id']] = row['cnt'] # Orders order_data = {} for start in range(0, len(uid_list), batch_size): batch = uid_list[start:start+batch_size] placeholders = ','.join(['%s'] * len(batch)) cur.execute(f""" SELECT o.account_id, o.trade_no, o.pay_success_date::date as pay_date, o.key_from, o.pay_amount_int, o.order_status, o.out_trade_no FROM bi_vala_order o JOIN bi_vala_app_account a ON o.account_id = a.id AND a.status = 1 AND a.deleted_at IS NULL WHERE o.account_id IN ({placeholders}) AND o.pay_success_date IS NOT NULL AND o.order_status IN (3, 4) ORDER BY o.account_id, o.pay_success_date """, batch) for row in cur.fetchall(): uid = row['account_id'] if uid not in order_data: order_data[uid] = [] order_data[uid].append({ 'trade_no': row['trade_no'], 'pay_date': str(row['pay_date']) if row['pay_date'] else None, 'key_from': row['key_from'], 'pay_amount_int': row['pay_amount_int'], 'order_status': row['order_status'], 'out_trade_no': row['out_trade_no'] }) # Refunds all_trade_nos = [] for uid, orders in order_data.items(): for o in orders: all_trade_nos.append(o['trade_no']) refund_map = {} if all_trade_nos: for start in range(0, len(all_trade_nos), batch_size): batch = all_trade_nos[start:start+batch_size] placeholders = ','.join(['%s'] * len(batch)) cur.execute(f""" SELECT r.trade_no, r.refund_amount FROM bi_refund_order r JOIN bi_vala_order o ON r.trade_no = o.trade_no AND o.order_status = 4 WHERE r.trade_no IN ({placeholders}) AND r.status = 3 """, batch) for row in cur.fetchall(): refund_map[row['trade_no']] = int(float(row['refund_amount'])) if row['refund_amount'] else 0 cur.close() conn.close() # Build expected values col_letters = 'ABCDEFGHIJKLMNOPQRSTUV' expected = {} # (row_idx, col_idx) -> value for idx in range(len(rows)): if idx not in uid_map: continue info = uid_map[idx] uid = info['uid'] expected[(idx, 7)] = str(uid) # H tc = trial_counts.get(uid, 0) expected[(idx, 3)] = tc # D if info['created_at']: expected[(idx, 8)] = info['created_at'] # I if info['download_channel']: expected[(idx, 9)] = info['download_channel'] # J orders = order_data.get(uid, []) if orders: expected[(idx, 10)] = '是' # K first = orders[0] if first['pay_date']: expected[(idx, 11)] = first['pay_date'] # L if first['key_from']: expected[(idx, 12)] = first['key_from'] # M total_gmv = sum(o['pay_amount_int'] for o in orders) gmv_yuan = total_gmv / 100.0 if gmv_yuan > 0: expected[(idx, 14)] = gmv_yuan # O total_refund = sum(refund_map.get(o['trade_no'], 0) for o in orders) if total_refund > 0: expected[(idx, 15)] = total_refund # P gsv = gmv_yuan - total_refund expected[(idx, 16)] = gsv # Q # Find missing cells missing = [] for (idx, col), val in expected.items(): row_data = rows[idx] if idx < len(rows) else [] current = str(row_data[col]) if col < len(row_data) and row_data[col] is not None else '' current = current.strip() expected_str = str(val).strip() if current != expected_str: missing.append((idx, col, val)) print(f"Missing cells to write: {len(missing)}") # Write one by one with delays success = 0 fail = 0 token_refreshes = 0 for i, (row_idx, col_idx, val) in enumerate(missing): row_num = row_idx + 3 col = col_letters[col_idx] result = write_cell(TOKEN, row_num, col, val) if result == 'TOKEN_EXPIRED': TOKEN = get_token() token_refreshes += 1 print(f" Token refreshed ({token_refreshes})") result = write_cell(TOKEN, row_num, col, val) if result is True: success += 1 else: fail += 1 if (i + 1) % 100 == 0: print(f" Progress: {i+1}/{len(missing)} (success={success}, fail={fail})") time.sleep(0.15) # ~6-7 writes/sec to avoid rate limits print(f"\n=== Retry Complete ===") print(f"Success: {success}, Fail: {fail}") print(f"Token refreshes: {token_refreshes}")