FIX: Switch back to using a temp table for moving posts. (#30509)

This commit is contained in:
Gary Pendergast 2024-12-31 17:02:27 +11:00 committed by GitHub
parent 9b9babdd72
commit 25351bdb8e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -108,6 +108,7 @@ class PostMover
end end
end end
create_temp_table
move_each_post move_each_post
handle_moved_references handle_moved_references
@ -129,6 +130,24 @@ class PostMover
destination_topic destination_topic
end end
def create_temp_table
DB.exec("DROP TABLE IF EXISTS temp_moved_posts") if Rails.env.test?
DB.exec <<~SQL
CREATE TEMPORARY TABLE temp_moved_posts (
old_topic_id INTEGER,
old_post_id INTEGER,
old_post_number INTEGER,
new_topic_id INTEGER,
new_topic_title VARCHAR,
new_post_id INTEGER,
new_post_number INTEGER
) ON COMMIT DROP;
CREATE INDEX moved_posts_old_post_number ON temp_moved_posts(old_post_number);
CREATE INDEX moved_posts_old_post_id ON temp_moved_posts(old_post_id);
SQL
end
def handle_moved_references def handle_moved_references
move_incoming_emails move_incoming_emails
move_notifications move_notifications
@ -372,6 +391,10 @@ class PostMover
metadata[:user_id] = @user.id metadata[:user_id] = @user.id
metadata[:full_move] = @full_move metadata[:full_move] = @full_move
DB.exec(<<~SQL, metadata)
INSERT INTO temp_moved_posts(old_topic_id, old_post_id, old_post_number, new_topic_id, new_topic_title, new_post_id, new_post_number)
VALUES (:old_topic_id, :old_post_id, :old_post_number, :new_topic_id, :new_topic_title, :new_post_id, :new_post_number)
SQL
DB.exec(<<~SQL, metadata) DB.exec(<<~SQL, metadata)
INSERT INTO moved_posts(old_topic_id, old_topic_title, old_post_id, old_post_number, post_user_id, user_id, full_move, new_topic_id, new_topic_title, new_post_id, new_post_number, created_new_topic, created_at, updated_at) INSERT INTO moved_posts(old_topic_id, old_topic_title, old_post_id, old_post_number, post_user_id, user_id, full_move, new_topic_id, new_topic_title, new_post_id, new_post_number, created_new_topic, created_at, updated_at)
VALUES (:old_topic_id, :old_topic_title, :old_post_id, :old_post_number, :post_user_id, :user_id, :full_move, :new_topic_id, :new_topic_title, :new_post_id, :new_post_number, :created_new_topic, :now, :now) VALUES (:old_topic_id, :old_topic_title, :old_post_id, :old_post_number, :post_user_id, :user_id, :full_move, :new_topic_id, :new_topic_title, :new_post_id, :new_post_number, :created_new_topic, :now, :now)
@ -387,7 +410,7 @@ class PostMover
UPDATE incoming_emails ie UPDATE incoming_emails ie
SET topic_id = mp.new_topic_id, SET topic_id = mp.new_topic_id,
post_id = mp.new_post_id post_id = mp.new_post_id
FROM moved_posts mp FROM temp_moved_posts mp
WHERE ie.topic_id = mp.old_topic_id AND ie.post_id = mp.old_post_id WHERE ie.topic_id = mp.old_topic_id AND ie.post_id = mp.old_post_id
AND mp.old_topic_id <> mp.new_topic_id AND mp.old_topic_id <> mp.new_topic_id
SQL SQL
@ -410,7 +433,7 @@ class PostMover
ELSE mp.new_topic_title END ELSE mp.new_topic_title END
) )
)) :: JSON )) :: JSON
FROM moved_posts mp FROM temp_moved_posts mp
WHERE n.topic_id = mp.old_topic_id AND n.post_number = mp.old_post_number WHERE n.topic_id = mp.old_topic_id AND n.post_number = mp.old_post_number
AND n.notification_type <> #{Notification.types[:watching_first_post]} AND n.notification_type <> #{Notification.types[:watching_first_post]}
SQL SQL
@ -422,7 +445,7 @@ class PostMover
SET reply_count = GREATEST(0, reply_count - x.moved_reply_count) SET reply_count = GREATEST(0, reply_count - x.moved_reply_count)
FROM ( FROM (
SELECT r.post_id, mp.new_topic_id, COUNT(1) AS moved_reply_count SELECT r.post_id, mp.new_topic_id, COUNT(1) AS moved_reply_count
FROM moved_posts mp FROM temp_moved_posts mp
JOIN post_replies r ON (mp.old_post_id = r.reply_post_id) JOIN post_replies r ON (mp.old_post_id = r.reply_post_id)
GROUP BY r.post_id, mp.new_topic_id GROUP BY r.post_id, mp.new_topic_id
) x ) x
@ -437,7 +460,7 @@ class PostMover
', post:' || mp.old_post_number || ', topic:' || mp.old_topic_id, ', post:' || mp.old_post_number || ', topic:' || mp.old_topic_id,
', post:' || mp.new_post_number || ', topic:' || mp.new_topic_id), ', post:' || mp.new_post_number || ', topic:' || mp.new_topic_id),
baked_version = NULL baked_version = NULL
FROM moved_posts mp, quoted_posts qp FROM temp_moved_posts mp, quoted_posts qp
WHERE p.id = qp.post_id AND mp.old_post_id = qp.quoted_post_id WHERE p.id = qp.post_id AND mp.old_post_id = qp.quoted_post_id
SQL SQL
end end
@ -446,15 +469,15 @@ class PostMover
DB.exec <<~SQL DB.exec <<~SQL
UPDATE post_replies pr UPDATE post_replies pr
SET post_id = mp.new_post_id SET post_id = mp.new_post_id
FROM moved_posts mp FROM temp_moved_posts mp
WHERE mp.old_post_id <> mp.new_post_id AND pr.post_id = mp.old_post_id AND WHERE mp.old_post_id <> mp.new_post_id AND pr.post_id = mp.old_post_id AND
EXISTS (SELECT 1 FROM moved_posts mr WHERE mr.new_post_id = pr.reply_post_id) EXISTS (SELECT 1 FROM temp_moved_posts mr WHERE mr.new_post_id = pr.reply_post_id)
SQL SQL
end end
def delete_post_replies def delete_post_replies
DB.exec <<~SQL DB.exec <<~SQL
DELETE FROM post_replies pr USING moved_posts mp DELETE FROM post_replies pr USING temp_moved_posts mp
WHERE (SELECT topic_id FROM posts WHERE id = pr.post_id) <> WHERE (SELECT topic_id FROM posts WHERE id = pr.post_id) <>
(SELECT topic_id FROM posts WHERE id = pr.reply_post_id) (SELECT topic_id FROM posts WHERE id = pr.reply_post_id)
AND (pr.reply_post_id = mp.old_post_id OR pr.post_id = mp.old_post_id) AND (pr.reply_post_id = mp.old_post_id OR pr.post_id = mp.old_post_id)
@ -471,7 +494,7 @@ class PostMover
AS ( AS (
SELECT pt.topic_id, mp.new_post_number as post_number, pt.user_id, pt.msecs SELECT pt.topic_id, mp.new_post_number as post_number, pt.user_id, pt.msecs
FROM post_timings pt FROM post_timings pt
JOIN moved_posts mp JOIN temp_moved_posts mp
ON mp.old_topic_id = pt.topic_id ON mp.old_topic_id = pt.topic_id
AND mp.old_post_number = pt.post_number AND mp.old_post_number = pt.post_number
AND mp.old_topic_id = mp.new_topic_id AND mp.old_topic_id = mp.new_topic_id
@ -495,7 +518,7 @@ class PostMover
INSERT INTO post_timings (topic_id, user_id, post_number, msecs) INSERT INTO post_timings (topic_id, user_id, post_number, msecs)
SELECT mp.new_topic_id, pt.user_id, mp.new_post_number, pt.msecs SELECT mp.new_topic_id, pt.user_id, mp.new_post_number, pt.msecs
FROM post_timings pt FROM post_timings pt
JOIN moved_posts mp ON (pt.topic_id = mp.old_topic_id AND pt.post_number = mp.old_post_number) JOIN temp_moved_posts mp ON (pt.topic_id = mp.old_topic_id AND pt.post_number = mp.old_post_number)
WHERE mp.old_post_id <> mp.new_post_id WHERE mp.old_post_id <> mp.new_post_id
AND mp.old_post_id IN (:post_ids) AND mp.old_post_id IN (:post_ids)
ON CONFLICT (topic_id, post_number, user_id) DO UPDATE ON CONFLICT (topic_id, post_number, user_id) DO UPDATE
@ -507,7 +530,7 @@ class PostMover
DB.exec <<~SQL DB.exec <<~SQL
DELETE DELETE
FROM post_timings pt FROM post_timings pt
USING moved_posts mp USING temp_moved_posts mp
WHERE pt.topic_id = mp.new_topic_id WHERE pt.topic_id = mp.new_topic_id
AND pt.post_number = mp.new_post_number AND pt.post_number = mp.new_post_number
SQL SQL
@ -518,7 +541,7 @@ class PostMover
UPDATE post_timings pt UPDATE post_timings pt
SET topic_id = mp.new_topic_id, SET topic_id = mp.new_topic_id,
post_number = mp.new_post_number post_number = mp.new_post_number
FROM moved_posts mp FROM temp_moved_posts mp
WHERE pt.topic_id = mp.old_topic_id WHERE pt.topic_id = mp.old_topic_id
AND pt.post_number = mp.old_post_number AND pt.post_number = mp.old_post_number
AND mp.old_post_id = mp.new_post_id AND mp.old_post_id = mp.new_post_id
@ -550,14 +573,14 @@ class PostMover
) AS posted, ) AS posted,
( (
SELECT MAX(lr.new_post_number) SELECT MAX(lr.new_post_number)
FROM moved_posts lr FROM temp_moved_posts lr
WHERE lr.old_topic_id = tu.topic_id WHERE lr.old_topic_id = tu.topic_id
AND lr.old_post_number <= tu.last_read_post_number AND lr.old_post_number <= tu.last_read_post_number
AND lr.old_topic_id <> lr.new_topic_id AND lr.old_topic_id <> lr.new_topic_id
) AS last_read_post_number, ) AS last_read_post_number,
( (
SELECT MAX(le.new_post_number) SELECT MAX(le.new_post_number)
FROM moved_posts le FROM temp_moved_posts le
WHERE le.old_topic_id = tu.topic_id WHERE le.old_topic_id = tu.topic_id
AND le.old_post_number <= tu.last_emailed_post_number AND le.old_post_number <= tu.last_emailed_post_number
AND le.old_topic_id <> le.new_topic_id AND le.old_topic_id <> le.new_topic_id
@ -573,7 +596,7 @@ class PostMover
AND GREATEST( AND GREATEST(
tu.last_read_post_number, tu.last_read_post_number,
tu.last_emailed_post_number tu.last_emailed_post_number
) >= (SELECT MIN(mp.old_post_number) FROM moved_posts mp WHERE mp.old_topic_id <> mp.new_topic_id) ) >= (SELECT MIN(mp.old_post_number) FROM temp_moved_posts mp WHERE mp.old_topic_id <> mp.new_topic_id)
ON CONFLICT (topic_id, user_id) DO UPDATE ON CONFLICT (topic_id, user_id) DO UPDATE
SET posted = excluded.posted, SET posted = excluded.posted,
last_read_post_number = CASE last_read_post_number = CASE