mirror of
https://github.com/1f349/dendrite.git
synced 2025-01-22 15:26:29 +00:00
61341aca50
... as they could fail if there are duplicate events in `roomserver_previous_events`. This fixes the migration by trying to combine the `event_nids` if possible (same room) as mentioned by @kegsay in https://github.com/matrix-org/dendrite/pull/3083#discussion_r1195508963
147 lines
4.8 KiB
Go
147 lines
4.8 KiB
Go
// Copyright 2023 The Matrix.org Foundation C.I.C.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package deltas
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"fmt"
|
|
|
|
"github.com/lib/pq"
|
|
"github.com/matrix-org/dendrite/internal"
|
|
"github.com/matrix-org/util"
|
|
)
|
|
|
|
func UpDropEventReferenceSHA(ctx context.Context, tx *sql.Tx) error {
|
|
var count int
|
|
err := tx.QueryRowContext(ctx, `SELECT count(*) FROM roomserver_events GROUP BY event_id HAVING count(event_id) > 1`).
|
|
Scan(&count)
|
|
if err != nil && err != sql.ErrNoRows {
|
|
return fmt.Errorf("failed to query duplicate event ids")
|
|
}
|
|
if count > 0 {
|
|
return fmt.Errorf("unable to drop column, as there are duplicate event ids")
|
|
}
|
|
_, err = tx.ExecContext(ctx, `ALTER TABLE roomserver_events DROP COLUMN reference_sha256;`)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to execute upgrade: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func UpDropEventReferenceSHAPrevEvents(ctx context.Context, tx *sql.Tx) error {
|
|
// rename the table
|
|
if _, err := tx.ExecContext(ctx, `ALTER TABLE roomserver_previous_events RENAME TO _roomserver_previous_events;`); err != nil {
|
|
return fmt.Errorf("tx.ExecContext: %w", err)
|
|
}
|
|
|
|
// create new table
|
|
if _, err := tx.ExecContext(ctx, `CREATE TABLE IF NOT EXISTS roomserver_previous_events (
|
|
previous_event_id TEXT NOT NULL,
|
|
event_nids TEXT NOT NULL,
|
|
UNIQUE (previous_event_id)
|
|
);`); err != nil {
|
|
return fmt.Errorf("tx.ExecContext: %w", err)
|
|
}
|
|
|
|
// figure out if there are duplicates
|
|
dupeRows, err := tx.QueryContext(ctx, `SELECT previous_event_id FROM _roomserver_previous_events GROUP BY previous_event_id HAVING count(previous_event_id) > 1`)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to query duplicate event ids")
|
|
}
|
|
defer internal.CloseAndLogIfError(ctx, dupeRows, "failed to close rows")
|
|
|
|
var prevEvents []string
|
|
var prevEventID string
|
|
for dupeRows.Next() {
|
|
if err = dupeRows.Scan(&prevEventID); err != nil {
|
|
return err
|
|
}
|
|
prevEvents = append(prevEvents, prevEventID)
|
|
}
|
|
if dupeRows.Err() != nil {
|
|
return dupeRows.Err()
|
|
}
|
|
|
|
// if we found duplicates, check if we can combine them, e.g. they are in the same room
|
|
for _, dupeID := range prevEvents {
|
|
var dupeNIDsRows *sql.Rows
|
|
dupeNIDsRows, err = tx.QueryContext(ctx, `SELECT event_nids FROM _roomserver_previous_events WHERE previous_event_id = $1`, dupeID)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to query duplicate event ids")
|
|
}
|
|
defer internal.CloseAndLogIfError(ctx, dupeNIDsRows, "failed to close rows")
|
|
var dupeNIDs []int64
|
|
for dupeNIDsRows.Next() {
|
|
var nids pq.Int64Array
|
|
if err = dupeNIDsRows.Scan(&nids); err != nil {
|
|
return err
|
|
}
|
|
dupeNIDs = append(dupeNIDs, nids...)
|
|
}
|
|
|
|
if dupeNIDsRows.Err() != nil {
|
|
return dupeNIDsRows.Err()
|
|
}
|
|
// dedupe NIDs
|
|
dupeNIDs = dupeNIDs[:util.SortAndUnique(nids(dupeNIDs))]
|
|
// now that we have all NIDs, check which room they belong to
|
|
var roomCount int
|
|
err = tx.QueryRowContext(ctx, `SELECT count(distinct room_nid) FROM roomserver_events WHERE event_nid IN ($1)`, pq.Array(dupeNIDs)).Scan(&roomCount)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// if the events are from different rooms, that's bad and we can't continue
|
|
if roomCount > 1 {
|
|
return fmt.Errorf("detected events (%v) referenced for different rooms (%v)", dupeNIDs, roomCount)
|
|
}
|
|
// otherwise delete the dupes
|
|
_, err = tx.ExecContext(ctx, "DELETE FROM _roomserver_previous_events WHERE previous_event_id = $1", dupeID)
|
|
if err != nil {
|
|
return fmt.Errorf("unable to delete duplicates: %w", err)
|
|
}
|
|
|
|
// insert combined values
|
|
_, err = tx.ExecContext(ctx, "INSERT INTO _roomserver_previous_events (previous_event_id, event_nids) VALUES ($1, $2)", dupeID, pq.Array(dupeNIDs))
|
|
if err != nil {
|
|
return fmt.Errorf("unable to insert new event NIDs: %w", err)
|
|
}
|
|
}
|
|
|
|
// move data
|
|
if _, err = tx.ExecContext(ctx, `
|
|
INSERT
|
|
INTO roomserver_previous_events (
|
|
previous_event_id, event_nids
|
|
) SELECT
|
|
previous_event_id, event_nids
|
|
FROM _roomserver_previous_events
|
|
;`); err != nil {
|
|
return fmt.Errorf("tx.ExecContext: %w", err)
|
|
}
|
|
// drop old table
|
|
_, err = tx.ExecContext(ctx, `DROP TABLE _roomserver_previous_events;`)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to execute upgrade: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type nids []int64
|
|
|
|
func (s nids) Len() int { return len(s) }
|
|
func (s nids) Less(i, j int) bool { return s[i] < s[j] }
|
|
func (s nids) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
|