2021-09-19 13:49:59 +02:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2022-11-27 19:20:29 +01:00
// SPDX-License-Identifier: MIT
2021-09-19 13:49:59 +02:00
package db
import (
2022-11-19 09:12:33 +01:00
"context"
2021-09-19 13:49:59 +02:00
"fmt"
"regexp"
"code.gitea.io/gitea/modules/setting"
)
// CountBadSequences looks for broken sequences from recreate-table mistakes
2022-11-19 09:12:33 +01:00
func CountBadSequences ( _ context . Context ) ( int64 , error ) {
2023-03-07 11:51:06 +01:00
if ! setting . Database . Type . IsPostgreSQL ( ) {
2021-09-19 13:49:59 +02:00
return 0 , nil
}
2024-12-20 17:11:38 +01:00
sess := xormEngine . NewSession ( )
2021-09-19 13:49:59 +02:00
defer sess . Close ( )
var sequences [ ] string
2024-12-20 17:11:38 +01:00
schema := xormEngine . Dialect ( ) . URI ( ) . Schema
2021-09-19 13:49:59 +02:00
sess . Engine ( ) . SetSchema ( "" )
if err := sess . Table ( "information_schema.sequences" ) . Cols ( "sequence_name" ) . Where ( "sequence_name LIKE 'tmp_recreate__%_id_seq%' AND sequence_catalog = ?" , setting . Database . Name ) . Find ( & sequences ) ; err != nil {
return 0 , err
}
sess . Engine ( ) . SetSchema ( schema )
return int64 ( len ( sequences ) ) , nil
}
// FixBadSequences fixes for broken sequences from recreate-table mistakes
2022-11-19 09:12:33 +01:00
func FixBadSequences ( _ context . Context ) error {
2023-03-07 11:51:06 +01:00
if ! setting . Database . Type . IsPostgreSQL ( ) {
2021-09-19 13:49:59 +02:00
return nil
}
2024-12-20 17:11:38 +01:00
sess := xormEngine . NewSession ( )
2021-09-19 13:49:59 +02:00
defer sess . Close ( )
if err := sess . Begin ( ) ; err != nil {
return err
}
var sequences [ ] string
schema := sess . Engine ( ) . Dialect ( ) . URI ( ) . Schema
sess . Engine ( ) . SetSchema ( "" )
if err := sess . Table ( "information_schema.sequences" ) . Cols ( "sequence_name" ) . Where ( "sequence_name LIKE 'tmp_recreate__%_id_seq%' AND sequence_catalog = ?" , setting . Database . Name ) . Find ( & sequences ) ; err != nil {
return err
}
sess . Engine ( ) . SetSchema ( schema )
sequenceRegexp := regexp . MustCompile ( ` tmp_recreate__(\w+)_id_seq.* ` )
for _ , sequence := range sequences {
tableName := sequenceRegexp . FindStringSubmatch ( sequence ) [ 1 ]
newSequenceName := tableName + "_id_seq"
if _ , err := sess . Exec ( fmt . Sprintf ( "ALTER SEQUENCE `%s` RENAME TO `%s`" , sequence , newSequenceName ) ) ; err != nil {
return err
}
if _ , err := sess . Exec ( fmt . Sprintf ( "SELECT setval('%s', COALESCE((SELECT MAX(id)+1 FROM `%s`), 1), false)" , newSequenceName , tableName ) ) ; err != nil {
return err
}
}
return sess . Commit ( )
}