2023-05-24 11:14:42 +01:00
// 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"
2023-05-30 17:05:48 +01:00
"github.com/lib/pq"
"github.com/matrix-org/dendrite/internal"
"github.com/matrix-org/util"
2023-05-24 11:14:42 +01:00
)
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 )
}
2023-05-30 17:05:48 +01:00
// 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 )
}
}
2023-05-24 11:14:42 +01:00
// move data
2023-05-30 17:05:48 +01:00
if _ , err = tx . ExecContext ( ctx , `
2023-05-24 11:14:42 +01:00
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
2023-05-30 17:05:48 +01:00
_ , err = tx . ExecContext ( ctx , ` DROP TABLE _roomserver_previous_events; ` )
2023-05-24 11:14:42 +01:00
if err != nil {
return fmt . Errorf ( "failed to execute upgrade: %w" , err )
}
return nil
}
2023-05-30 17:05:48 +01:00
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 ] }