2024-01-12 13:15:30 +08:00
package postgres
2022-10-15 17:38:34 +08:00
import (
"fmt"
2024-01-12 13:15:30 +08:00
"mayfly-go/internal/db/dbm/dbi"
2022-12-22 18:41:34 +08:00
"time"
2024-03-21 17:15:52 +08:00
2025-06-27 12:17:45 +08:00
"github.com/spf13/cast"
2022-10-15 17:38:34 +08:00
)
2023-11-26 21:21:35 +08:00
type PgsqlDialect struct {
2024-03-18 12:25:40 +08:00
dbi . DefaultDialect
2022-10-15 17:38:34 +08:00
2024-03-18 12:25:40 +08:00
dc * dbi . DbConn
2024-01-05 08:55:34 +08:00
}
2024-01-05 05:31:32 +00:00
2024-03-01 04:03:03 +00:00
func ( pd * PgsqlDialect ) CopyTable ( copy * dbi . DbCopyTable ) error {
2024-01-23 04:08:02 +00:00
tableName := copy . TableName
// 生成新表名,为老表明+_copy_时间戳
newTableName := tableName + "_copy_" + time . Now ( ) . Format ( "20060102150405" )
// 执行根据旧表创建新表
2024-03-01 04:03:03 +00:00
_ , err := pd . dc . Exec ( fmt . Sprintf ( "create table %s (like %s)" , newTableName , tableName ) )
2024-01-23 04:08:02 +00:00
if err != nil {
return err
}
// 复制数据
if copy . CopyData {
go func ( ) {
2024-03-01 04:03:03 +00:00
_ , _ = pd . dc . Exec ( fmt . Sprintf ( "insert into %s select * from %s" , newTableName , tableName ) )
2024-01-23 04:08:02 +00:00
} ( )
}
// 查询旧表的自增字段名 重新设置新表的序列序列器
2024-03-01 04:03:03 +00:00
_ , res , err := pd . dc . Query ( fmt . Sprintf ( "select column_name from information_schema.columns where table_name = '%s' and column_default like 'nextval%%'" , tableName ) )
2024-01-23 04:08:02 +00:00
if err != nil {
return err
}
for _ , re := range res {
2024-03-21 17:15:52 +08:00
colName := cast . ToString ( re [ "column_name" ] )
2024-01-23 04:08:02 +00:00
if colName != "" {
// 查询自增列当前最大值
2024-03-01 04:03:03 +00:00
_ , maxRes , err := pd . dc . Query ( fmt . Sprintf ( "select max(%s) max_val from %s" , colName , tableName ) )
2024-01-23 04:08:02 +00:00
if err != nil {
return err
}
2024-03-21 17:15:52 +08:00
maxVal := cast . ToInt ( maxRes [ 0 ] [ "max_val" ] )
2024-01-23 04:08:02 +00:00
// 序列起始值为1或当前最大值+1
if maxVal <= 0 {
maxVal = 1
} else {
maxVal += 1
}
// 之所以不用tableName_colName_seq是因为gauss会自动创建同名的序列, 且无法修改序列起始值, 所以直接使用新序列值
newSeqName := fmt . Sprintf ( "%s_%s_copy_seq" , newTableName , colName )
// 创建自增序列,当前最大值为旧表最大值
2024-03-01 04:03:03 +00:00
_ , err = pd . dc . Exec ( fmt . Sprintf ( "CREATE SEQUENCE %s START %d INCREMENT 1" , newSeqName , maxVal ) )
2024-01-23 04:08:02 +00:00
if err != nil {
return err
}
// 将新表的自增主键序列与主键列相关联
2024-03-01 04:03:03 +00:00
_ , err = pd . dc . Exec ( fmt . Sprintf ( "alter table %s alter column %s set default nextval('%s')" , newTableName , colName , newSeqName ) )
2024-01-23 04:08:02 +00:00
if err != nil {
return err
}
}
}
return err
}
2024-03-15 09:01:51 +00:00
2024-11-01 17:27:22 +08:00
func ( pd * PgsqlDialect ) GetDumpHelper ( ) dbi . DumpHelper {
return new ( DumpHelper )
}
2024-12-08 13:04:23 +08:00
func ( md * PgsqlDialect ) GetSQLGenerator ( ) dbi . SQLGenerator {
return & SQLGenerator {
dialect : md ,
dc : md . dc ,
2024-11-01 17:27:22 +08:00
}
2024-10-16 17:24:50 +08:00
}