2014-02-12 12:49:46 -05:00
// Copyright 2014 The Gogs Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package models
2014-02-13 23:23:23 +08:00
2014-02-18 17:48:02 -05:00
import (
2017-02-26 04:37:05 -05:00
"bufio"
2014-10-19 01:35:24 -04:00
"database/sql"
2016-08-11 14:38:26 -07:00
"errors"
2014-02-18 17:48:02 -05:00
"fmt"
2015-08-29 12:08:37 +08:00
"net/url"
2014-02-18 17:48:02 -05:00
"os"
2014-03-21 01:09:22 -04:00
"path"
2014-04-14 14:49:50 +08:00
"strings"
2019-02-06 18:46:15 -05:00
"time"
2014-02-18 17:48:02 -05:00
2017-02-26 04:37:05 -05:00
"github.com/Unknwon/com"
2017-02-14 02:50:00 +01:00
_ "github.com/denisenkom/go-mssqldb"
2014-02-18 17:48:02 -05:00
_ "github.com/go-sql-driver/mysql"
2015-01-23 09:54:16 +02:00
"github.com/go-xorm/core"
2014-04-18 09:35:09 -04:00
"github.com/go-xorm/xorm"
2018-06-09 17:21:58 +08:00
"github.com/json-iterator/go"
2014-03-17 14:03:58 -04:00
_ "github.com/lib/pq"
2017-03-23 18:34:25 -04:00
log "gopkg.in/clog.v1"
2014-02-18 17:48:02 -05:00
2018-05-27 08:53:48 +08:00
"github.com/gogs/gogs/models/migrations"
"github.com/gogs/gogs/pkg/setting"
2014-02-18 17:48:02 -05:00
)
2014-02-13 23:23:23 +08:00
2017-04-06 00:14:30 -04:00
// Engine represents a XORM engine or session.
2014-10-19 01:35:24 -04:00
type Engine interface {
Delete ( interface { } ) ( int64 , error )
Exec ( string , ... interface { } ) ( sql . Result , error )
2015-02-13 00:58:46 -05:00
Find ( interface { } , ... interface { } ) error
2015-02-10 23:44:16 -05:00
Get ( interface { } ) ( bool , error )
2018-08-16 20:26:09 +08:00
ID ( interface { } ) * xorm . Session
2016-08-15 18:40:32 -07:00
In ( string , ... interface { } ) * xorm . Session
2014-10-19 01:35:24 -04:00
Insert ( ... interface { } ) ( int64 , error )
2015-02-13 00:58:46 -05:00
InsertOne ( interface { } ) ( int64 , error )
2016-07-26 17:26:48 +08:00
Iterate ( interface { } , xorm . IterFunc ) error
2015-02-13 00:58:46 -05:00
Sql ( string , ... interface { } ) * xorm . Session
2016-12-20 23:23:57 -05:00
Table ( interface { } ) * xorm . Session
2016-09-23 07:38:12 +08:00
Where ( interface { } , ... interface { } ) * xorm . Session
2014-10-19 01:35:24 -04:00
}
2014-03-21 01:48:10 -04:00
var (
2014-10-19 01:35:24 -04:00
x * xorm . Engine
tables [ ] interface { }
2014-03-30 10:47:08 -04:00
HasEngine bool
2014-03-21 01:48:10 -04:00
2014-03-21 03:27:59 -04:00
DbCfg struct {
2015-02-01 12:41:03 -05:00
Type , Host , Name , User , Passwd , Path , SSLMode string
2014-03-21 01:48:10 -04:00
}
2014-03-30 16:01:50 -04:00
2014-04-12 16:24:09 -04:00
EnableSQLite3 bool
2014-03-21 01:48:10 -04:00
)
2014-04-05 22:46:32 +08:00
func init ( ) {
2014-11-12 06:48:50 -05:00
tables = append ( tables ,
2017-04-06 00:14:30 -04:00
new ( User ) , new ( PublicKey ) , new ( AccessToken ) , new ( TwoFactor ) , new ( TwoFactorRecoveryCode ) ,
2016-08-30 05:07:50 -07:00
new ( Repository ) , new ( DeployKey ) , new ( Collaboration ) , new ( Access ) , new ( Upload ) ,
2015-09-01 19:07:02 -04:00
new ( Watch ) , new ( Star ) , new ( Follow ) , new ( Action ) ,
2015-09-02 05:09:12 -04:00
new ( Issue ) , new ( PullRequest ) , new ( Comment ) , new ( Attachment ) , new ( IssueUser ) ,
2015-08-10 14:42:50 +08:00
new ( Label ) , new ( IssueLabel ) , new ( Milestone ) ,
2017-02-17 15:10:50 -05:00
new ( Mirror ) , new ( Release ) , new ( LoginSource ) , new ( Webhook ) , new ( HookTask ) ,
2017-02-23 18:25:12 -05:00
new ( ProtectBranch ) , new ( ProtectBranchWhitelist ) ,
2015-02-23 02:15:53 -05:00
new ( Team ) , new ( OrgUser ) , new ( TeamUser ) , new ( TeamRepo ) ,
2015-02-11 21:58:37 -05:00
new ( Notice ) , new ( EmailAddress ) )
2015-08-27 23:06:14 +08:00
2015-11-07 00:39:45 -05:00
gonicNames := [ ] string { "SSL" }
2015-08-27 23:06:14 +08:00
for _ , name := range gonicNames {
core . LintGonicMapper [ name ] = true
}
2014-04-05 22:46:32 +08:00
}
2015-09-16 23:08:46 -04:00
func LoadConfigs ( ) {
2014-12-31 18:37:29 +08:00
sec := setting . Cfg . Section ( "database" )
DbCfg . Type = sec . Key ( "DB_TYPE" ) . String ( )
2015-02-11 21:58:37 -05:00
switch DbCfg . Type {
case "sqlite3" :
setting . UseSQLite3 = true
case "mysql" :
setting . UseMySQL = true
case "postgres" :
setting . UsePostgreSQL = true
2017-02-14 02:50:00 +01:00
case "mssql" :
setting . UseMSSQL = true
2014-03-30 16:01:50 -04:00
}
2014-12-31 18:37:29 +08:00
DbCfg . Host = sec . Key ( "HOST" ) . String ( )
DbCfg . Name = sec . Key ( "NAME" ) . String ( )
DbCfg . User = sec . Key ( "USER" ) . String ( )
2015-02-01 12:41:03 -05:00
if len ( DbCfg . Passwd ) == 0 {
DbCfg . Passwd = sec . Key ( "PASSWD" ) . String ( )
2014-06-10 19:11:53 -04:00
}
2015-02-01 12:41:03 -05:00
DbCfg . SSLMode = sec . Key ( "SSL_MODE" ) . String ( )
2014-12-31 18:37:29 +08:00
DbCfg . Path = sec . Key ( "PATH" ) . MustString ( "data/gogs.db" )
2014-03-21 01:48:10 -04:00
}
2014-02-18 17:48:02 -05:00
2016-08-12 02:56:50 -07:00
// parsePostgreSQLHostPort parses given input in various forms defined in
// https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-CONNSTRING
// and returns proper host and port number.
func parsePostgreSQLHostPort ( info string ) ( string , string ) {
host , port := "127.0.0.1" , "5432"
if strings . Contains ( info , ":" ) && ! strings . HasSuffix ( info , "]" ) {
idx := strings . LastIndex ( info , ":" )
host = info [ : idx ]
port = info [ idx + 1 : ]
} else if len ( info ) > 0 {
host = info
}
return host , port
}
2017-02-14 02:50:00 +01:00
func parseMSSQLHostPort ( info string ) ( string , string ) {
host , port := "127.0.0.1" , "1433"
if strings . Contains ( info , ":" ) {
host = strings . Split ( info , ":" ) [ 0 ]
port = strings . Split ( info , ":" ) [ 1 ]
} else if strings . Contains ( info , "," ) {
host = strings . Split ( info , "," ) [ 0 ]
port = strings . TrimSpace ( strings . Split ( info , "," ) [ 1 ] )
} else if len ( info ) > 0 {
host = info
}
return host , port
}
2014-09-04 17:19:26 +02:00
func getEngine ( ) ( * xorm . Engine , error ) {
2016-08-11 14:38:26 -07:00
connStr := ""
2016-07-02 10:39:39 -04:00
var Param string = "?"
2016-07-24 14:32:46 +08:00
if strings . Contains ( DbCfg . Name , Param ) {
Param = "&"
2016-07-02 10:39:39 -04:00
}
2014-03-30 10:47:08 -04:00
switch DbCfg . Type {
case "mysql" :
2015-03-14 02:21:47 +08:00
if DbCfg . Host [ 0 ] == '/' { // looks like a unix socket
2017-11-23 02:42:30 +08:00
connStr = fmt . Sprintf ( "%s:%s@unix(%s)/%s%scharset=utf8mb4&parseTime=true" ,
2016-07-02 10:39:39 -04:00
DbCfg . User , DbCfg . Passwd , DbCfg . Host , DbCfg . Name , Param )
2015-03-14 02:21:47 +08:00
} else {
2017-11-23 02:42:30 +08:00
connStr = fmt . Sprintf ( "%s:%s@tcp(%s)/%s%scharset=utf8mb4&parseTime=true" ,
2016-07-02 10:39:39 -04:00
DbCfg . User , DbCfg . Passwd , DbCfg . Host , DbCfg . Name , Param )
2015-03-14 02:21:47 +08:00
}
2018-03-10 04:44:07 +08:00
var engineParams = map [ string ] string { "rowFormat" : "DYNAMIC" }
return xorm . NewEngineWithParams ( DbCfg . Type , connStr , engineParams )
2014-03-30 10:47:08 -04:00
case "postgres" :
2016-08-12 03:04:50 -07:00
host , port := parsePostgreSQLHostPort ( DbCfg . Host )
2016-08-11 14:38:26 -07:00
if host [ 0 ] == '/' { // looks like a unix socket
connStr = fmt . Sprintf ( "postgres://%s:%s@:%s/%s%ssslmode=%s&host=%s" ,
url . QueryEscape ( DbCfg . User ) , url . QueryEscape ( DbCfg . Passwd ) , port , DbCfg . Name , Param , DbCfg . SSLMode , host )
} else {
connStr = fmt . Sprintf ( "postgres://%s:%s@%s:%s/%s%ssslmode=%s" ,
url . QueryEscape ( DbCfg . User ) , url . QueryEscape ( DbCfg . Passwd ) , host , port , DbCfg . Name , Param , DbCfg . SSLMode )
}
2017-02-14 02:50:00 +01:00
case "mssql" :
host , port := parseMSSQLHostPort ( DbCfg . Host )
connStr = fmt . Sprintf ( "server=%s; port=%s; database=%s; user id=%s; password=%s;" , host , port , DbCfg . Name , DbCfg . User , DbCfg . Passwd )
2014-04-12 11:48:12 -07:00
case "sqlite3" :
2014-04-12 16:24:09 -04:00
if ! EnableSQLite3 {
2016-08-11 14:38:26 -07:00
return nil , errors . New ( "This binary version does not build support for SQLite3." )
2014-04-12 16:24:09 -04:00
}
2015-08-24 21:01:23 +08:00
if err := os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm ) ; err != nil {
return nil , fmt . Errorf ( "Fail to create directories: %v" , err )
}
2016-08-11 14:38:26 -07:00
connStr = "file:" + DbCfg . Path + "?cache=shared&mode=rwc"
2014-03-30 10:47:08 -04:00
default :
2014-09-04 17:19:26 +02:00
return nil , fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-03-30 10:47:08 -04:00
}
2016-08-11 14:38:26 -07:00
return xorm . NewEngine ( DbCfg . Type , connStr )
2014-09-04 17:19:26 +02:00
}
func NewTestEngine ( x * xorm . Engine ) ( err error ) {
x , err = getEngine ( )
2014-03-30 10:47:08 -04:00
if err != nil {
2015-08-02 12:36:35 +08:00
return fmt . Errorf ( "Connect to database: %v" , err )
2014-03-30 10:47:08 -04:00
}
2014-09-04 17:19:26 +02:00
2015-01-23 09:54:16 +02:00
x . SetMapper ( core . GonicMapper { } )
2015-09-03 05:05:58 -04:00
return x . StoreEngine ( "InnoDB" ) . Sync2 ( tables ... )
2014-03-30 10:47:08 -04:00
}
2014-03-29 17:50:51 -04:00
func SetEngine ( ) ( err error ) {
2014-09-04 17:19:26 +02:00
x , err = getEngine ( )
2014-02-18 17:48:02 -05:00
if err != nil {
2015-08-06 22:48:11 +08:00
return fmt . Errorf ( "Fail to connect to database: %v" , err )
2014-02-18 17:48:02 -05:00
}
2015-01-23 09:54:16 +02:00
x . SetMapper ( core . GonicMapper { } )
2014-12-06 20:22:48 -05:00
// WARNING: for serv command, MUST remove the output to os.stdout,
2014-03-20 16:04:56 -04:00
// so use log file to instead print to stdout.
2017-03-23 18:34:25 -04:00
sec := setting . Cfg . Section ( "log.xorm" )
logger , err := log . NewFileWriter ( path . Join ( setting . LogRootPath , "xorm.log" ) ,
log . FileRotationConfig {
Rotate : sec . Key ( "ROTATE" ) . MustBool ( true ) ,
Daily : sec . Key ( "ROTATE_DAILY" ) . MustBool ( true ) ,
MaxSize : sec . Key ( "MAX_SIZE" ) . MustInt64 ( 100 ) * 1024 * 1024 ,
MaxDays : sec . Key ( "MAX_DAYS" ) . MustInt64 ( 3 ) ,
} )
2014-03-17 14:03:58 -04:00
if err != nil {
2017-03-23 18:34:25 -04:00
return fmt . Errorf ( "Fail to create 'xorm.log': %v" , err )
2014-03-17 14:03:58 -04:00
}
2017-02-09 15:09:37 -05:00
2019-02-06 18:46:15 -05:00
// To prevent mystery "MySQL: invalid connection" error,
// see https://github.com/gogs/gogs/issues/5532.
x . SetMaxIdleConns ( 0 )
x . SetConnMaxLifetime ( time . Second )
2017-02-09 15:09:37 -05:00
if setting . ProdMode {
2017-03-23 18:34:25 -04:00
x . SetLogger ( xorm . NewSimpleLogger3 ( logger , xorm . DEFAULT_LOG_PREFIX , xorm . DEFAULT_LOG_FLAG , core . LOG_WARNING ) )
2017-02-09 15:09:37 -05:00
} else {
2017-03-23 18:34:25 -04:00
x . SetLogger ( xorm . NewSimpleLogger ( logger ) )
2017-02-09 15:09:37 -05:00
}
2016-02-16 22:35:08 +08:00
x . ShowSQL ( true )
2014-03-29 17:50:51 -04:00
return nil
2014-02-18 17:48:02 -05:00
}
2014-03-29 17:50:51 -04:00
func NewEngine ( ) ( err error ) {
if err = SetEngine ( ) ; err != nil {
return err
2014-04-05 22:46:32 +08:00
}
2015-01-22 14:49:52 +02:00
if err = migrations . Migrate ( x ) ; err != nil {
2015-02-11 21:58:37 -05:00
return fmt . Errorf ( "migrate: %v" , err )
2015-01-22 14:49:52 +02:00
}
2014-10-28 16:40:09 +01:00
if err = x . StoreEngine ( "InnoDB" ) . Sync2 ( tables ... ) ; err != nil {
2014-03-29 17:50:51 -04:00
return fmt . Errorf ( "sync database struct error: %v\n" , err )
2014-02-19 17:50:53 +08:00
}
2015-01-23 09:54:16 +02:00
2014-03-29 17:50:51 -04:00
return nil
2014-02-18 17:48:02 -05:00
}
2014-03-20 16:04:56 -04:00
type Statistic struct {
Counter struct {
2014-08-28 22:29:00 +08:00
User , Org , PublicKey ,
Repo , Watch , Star , Action , Access ,
Issue , Comment , Oauth , Follow ,
Mirror , Release , LoginSource , Webhook ,
Milestone , Label , HookTask ,
Team , UpdateTask , Attachment int64
2014-03-20 16:04:56 -04:00
}
}
func GetStatistic ( ) ( stats Statistic ) {
2014-07-07 04:15:08 -04:00
stats . Counter . User = CountUsers ( )
2014-08-28 22:29:00 +08:00
stats . Counter . Org = CountOrganizations ( )
2014-06-21 00:51:41 -04:00
stats . Counter . PublicKey , _ = x . Count ( new ( PublicKey ) )
2016-07-24 14:32:46 +08:00
stats . Counter . Repo = CountRepositories ( true )
2014-06-21 00:51:41 -04:00
stats . Counter . Watch , _ = x . Count ( new ( Watch ) )
2014-08-28 22:29:00 +08:00
stats . Counter . Star , _ = x . Count ( new ( Star ) )
2014-06-21 00:51:41 -04:00
stats . Counter . Action , _ = x . Count ( new ( Action ) )
stats . Counter . Access , _ = x . Count ( new ( Access ) )
stats . Counter . Issue , _ = x . Count ( new ( Issue ) )
stats . Counter . Comment , _ = x . Count ( new ( Comment ) )
2015-09-17 16:11:44 -04:00
stats . Counter . Oauth = 0
2014-08-28 22:29:00 +08:00
stats . Counter . Follow , _ = x . Count ( new ( Follow ) )
stats . Counter . Mirror , _ = x . Count ( new ( Mirror ) )
2014-06-21 00:51:41 -04:00
stats . Counter . Release , _ = x . Count ( new ( Release ) )
2015-09-10 15:45:03 -04:00
stats . Counter . LoginSource = CountLoginSources ( )
2014-06-21 00:51:41 -04:00
stats . Counter . Webhook , _ = x . Count ( new ( Webhook ) )
stats . Counter . Milestone , _ = x . Count ( new ( Milestone ) )
2014-08-28 22:29:00 +08:00
stats . Counter . Label , _ = x . Count ( new ( Label ) )
stats . Counter . HookTask , _ = x . Count ( new ( HookTask ) )
stats . Counter . Team , _ = x . Count ( new ( Team ) )
stats . Counter . Attachment , _ = x . Count ( new ( Attachment ) )
2014-03-23 16:31:13 +08:00
return
2014-03-20 16:04:56 -04:00
}
2014-05-05 00:55:17 -04:00
2014-08-06 17:21:24 -04:00
func Ping ( ) error {
return x . Ping ( )
}
2017-02-26 04:37:05 -05:00
// The version table. Should have only one row with id==1
type Version struct {
ID int64
Version int64
}
// DumpDatabase dumps all data from database to file system in JSON format.
func DumpDatabase ( dirPath string ) ( err error ) {
os . MkdirAll ( dirPath , os . ModePerm )
// Purposely create a local variable to not modify global variable
tables := append ( tables , new ( Version ) )
for _ , table := range tables {
tableName := strings . TrimPrefix ( fmt . Sprintf ( "%T" , table ) , "*models." )
tableFile := path . Join ( dirPath , tableName + ".json" )
f , err := os . Create ( tableFile )
if err != nil {
return fmt . Errorf ( "fail to create JSON file: %v" , err )
}
if err = x . Asc ( "id" ) . Iterate ( table , func ( idx int , bean interface { } ) ( err error ) {
2018-06-09 17:21:58 +08:00
return jsoniter . NewEncoder ( f ) . Encode ( bean )
2017-02-26 04:37:05 -05:00
} ) ; err != nil {
f . Close ( )
return fmt . Errorf ( "fail to dump table '%s': %v" , tableName , err )
}
f . Close ( )
}
return nil
}
// ImportDatabase imports data from backup archive.
2017-05-21 04:37:37 -04:00
func ImportDatabase ( dirPath string , verbose bool ) ( err error ) {
snakeMapper := core . SnakeMapper { }
2018-06-09 17:21:58 +08:00
skipInsertProcessors := map [ string ] bool {
2018-06-25 20:33:21 +08:00
"mirror" : true ,
"milestone" : true ,
2018-06-09 17:21:58 +08:00
}
2017-02-26 04:37:05 -05:00
// Purposely create a local variable to not modify global variable
tables := append ( tables , new ( Version ) )
for _ , table := range tables {
tableName := strings . TrimPrefix ( fmt . Sprintf ( "%T" , table ) , "*models." )
tableFile := path . Join ( dirPath , tableName + ".json" )
if ! com . IsExist ( tableFile ) {
continue
}
2017-05-21 04:37:37 -04:00
if verbose {
log . Trace ( "Importing table '%s'..." , tableName )
}
2017-02-26 04:37:05 -05:00
if err = x . DropTables ( table ) ; err != nil {
2018-06-09 17:21:58 +08:00
return fmt . Errorf ( "drop table '%s': %v" , tableName , err )
2017-02-26 04:37:05 -05:00
} else if err = x . Sync2 ( table ) ; err != nil {
2018-06-09 17:21:58 +08:00
return fmt . Errorf ( "sync table '%s': %v" , tableName , err )
2017-02-26 04:37:05 -05:00
}
f , err := os . Open ( tableFile )
if err != nil {
2018-06-09 17:21:58 +08:00
return fmt . Errorf ( "open JSON file: %v" , err )
2017-02-26 04:37:05 -05:00
}
2018-06-09 17:21:58 +08:00
rawTableName := x . TableName ( table )
_ , isInsertProcessor := table . ( xorm . BeforeInsertProcessor )
2017-02-26 04:37:05 -05:00
scanner := bufio . NewScanner ( f )
for scanner . Scan ( ) {
switch bean := table . ( type ) {
case * LoginSource :
meta := make ( map [ string ] interface { } )
2018-06-09 17:21:58 +08:00
if err = jsoniter . Unmarshal ( scanner . Bytes ( ) , & meta ) ; err != nil {
return fmt . Errorf ( "unmarshal to map: %v" , err )
2017-02-26 04:37:05 -05:00
}
tp := LoginType ( com . StrTo ( com . ToStr ( meta [ "Type" ] ) ) . MustInt64 ( ) )
switch tp {
case LOGIN_LDAP , LOGIN_DLDAP :
bean . Cfg = new ( LDAPConfig )
case LOGIN_SMTP :
bean . Cfg = new ( SMTPConfig )
case LOGIN_PAM :
bean . Cfg = new ( PAMConfig )
2018-12-18 12:49:30 -08:00
case LOGIN_GITHUB :
2018-12-18 16:46:50 -05:00
bean . Cfg = new ( GitHubConfig )
2017-02-26 04:37:05 -05:00
default :
return fmt . Errorf ( "unrecognized login source type:: %v" , tp )
}
table = bean
}
2018-06-09 17:21:58 +08:00
if err = jsoniter . Unmarshal ( scanner . Bytes ( ) , table ) ; err != nil {
return fmt . Errorf ( "unmarshal to struct: %v" , err )
2017-02-26 04:37:05 -05:00
}
if _ , err = x . Insert ( table ) ; err != nil {
2018-06-09 17:21:58 +08:00
return fmt . Errorf ( "insert strcut: %v" , err )
}
2018-06-27 22:08:24 +08:00
meta := make ( map [ string ] interface { } )
if err = jsoniter . Unmarshal ( scanner . Bytes ( ) , & meta ) ; err != nil {
log . Error ( 2 , "Failed to unmarshal to map: %v" , err )
}
2018-06-09 17:21:58 +08:00
// Reset created_unix back to the date save in archive because Insert method updates its value
if isInsertProcessor && ! skipInsertProcessors [ rawTableName ] {
if _ , err = x . Exec ( "UPDATE " + rawTableName + " SET created_unix=? WHERE id=?" , meta [ "CreatedUnix" ] , meta [ "ID" ] ) ; err != nil {
log . Error ( 2 , "Failed to reset 'created_unix': %v" , err )
}
2017-02-26 04:37:05 -05:00
}
2018-06-27 22:08:24 +08:00
switch rawTableName {
case "milestone" :
if _ , err = x . Exec ( "UPDATE " + rawTableName + " SET deadline_unix=?, closed_date_unix=? WHERE id=?" , meta [ "DeadlineUnix" ] , meta [ "ClosedDateUnix" ] , meta [ "ID" ] ) ; err != nil {
log . Error ( 2 , "Failed to reset 'milestone.deadline_unix', 'milestone.closed_date_unix': %v" , err )
}
}
2017-02-26 04:37:05 -05:00
}
2017-05-21 04:37:37 -04:00
// PostgreSQL needs manually reset table sequence for auto increment keys
if setting . UsePostgreSQL {
rawTableName := snakeMapper . Obj2Table ( tableName )
seqName := rawTableName + "_id_seq"
if _ , err = x . Exec ( fmt . Sprintf ( ` SELECT setval('%s', COALESCE((SELECT MAX(id)+1 FROM "%s"), 1), false); ` , seqName , rawTableName ) ) ; err != nil {
2018-06-09 17:21:58 +08:00
return fmt . Errorf ( "reset table '%s' sequence: %v" , rawTableName , err )
2017-05-21 04:37:37 -04:00
}
}
2017-02-26 04:37:05 -05:00
}
return nil
2014-05-05 00:55:17 -04:00
}