From e7b1e0c0edd4be2b628da59f8d20e555bde77a42 Mon Sep 17 00:00:00 2001 From: callebtc <93376500+callebtc@users.noreply.github.com> Date: Sat, 16 Mar 2024 10:56:51 +0100 Subject: [PATCH] Batch update Y columns in proofs_used and proofs_pending tables (#475) * mint: migration 16 batch update * migrate: only if data exists --- cashu/mint/migrations.py | 53 ++++++++++++++++++++++++++++++++++------ 1 file changed, 46 insertions(+), 7 deletions(-) diff --git a/cashu/mint/migrations.py b/cashu/mint/migrations.py index f215b2a2..2291288c 100644 --- a/cashu/mint/migrations.py +++ b/cashu/mint/migrations.py @@ -555,15 +555,54 @@ async def m016_recompute_Y_with_new_h2c(db: Database): ) proofs_pending = [Proof(**r) for r in rows] - # overwrite the old Y columns with the new Y + # Prepare data for batch update + proofs_used_data = [(proof.Y, proof.secret) for proof in proofs_used] + proofs_pending_data = [(proof.Y, proof.secret) for proof in proofs_pending] + + # Perform batch update in a single transaction async with db.connect() as conn: - for proof in proofs_used: + if len(proofs_used_data): + # For proofs_used await conn.execute( - f"UPDATE {table_with_schema(db, 'proofs_used')} SET Y = '{proof.Y}'" - f" WHERE secret = '{proof.secret}'" + "CREATE TABLE IF NOT EXISTS tmp_proofs_used (Y TEXT, secret TEXT)" + ) + values_placeholder = ", ".join( + f"('{y}', '{secret}')" for y, secret in proofs_used_data ) - for proof in proofs_pending: await conn.execute( - f"UPDATE {table_with_schema(db, 'proofs_pending')} SET Y = '{proof.Y}'" - f" WHERE secret = '{proof.secret}'" + f"INSERT INTO tmp_proofs_used (Y, secret) VALUES {values_placeholder}", + ) + await conn.execute( + f""" + UPDATE {table_with_schema(db, 'proofs_used')} + SET Y = tmp_proofs_used.Y + FROM tmp_proofs_used + WHERE {table_with_schema(db, 'proofs_used')}.secret = tmp_proofs_used.secret + """ ) + + if len(proofs_pending_data): + # For proofs_pending + await conn.execute( + "CREATE TABLE IF NOT EXISTS tmp_proofs_pending (Y TEXT, secret TEXT)" + ) + values_placeholder = ", ".join( + f"('{y}', '{secret}')" for y, secret in proofs_pending_data + ) + await conn.execute( + f"INSERT INTO tmp_proofs_used (Y, secret) VALUES {values_placeholder}", + ) + await conn.execute( + f""" + UPDATE {table_with_schema(db, 'proofs_pending')} + SET Y = tmp_proofs_pending.Y + FROM tmp_proofs_pending + WHERE {table_with_schema(db, 'proofs_pending')}.secret = tmp_proofs_pending.secret + """ + ) + + async with db.connect() as conn: + if len(proofs_used_data): + await conn.execute("DROP TABLE tmp_proofs_used") + if len(proofs_pending_data): + await conn.execute("DROP TABLE tmp_proofs_pending")