mirror of
https://gitee.com/johng/gf.git
synced 2024-11-29 18:57:44 +08:00
5e72b03b0a
* up * rename function names for package gtcp/gudp; add proxy example for gtcp.Server (#2295) * fix router supported for handler of package ghttp; fix json tag name issue when it contains for package goai * add proxy example for http server * rename function names for package gtcp/gudp; add proxy example for gtcp.Server * move TX from struct to interface for package gdb (#2247) * move TX from struct to interface for package gdb * i updates * up * up * fix comment Co-authored-by: houseme <housemecn@gmail.com> * move `go-redis` implements `Adapter` from package `gredis` to `contrib/nosql/redis`; add redis string operation functions for package `gredis` (#2240) * unify configuration pattern of for package gdb * version updates * improve implements `internal/rwmutex` and `internal/mutex`; add `TablesFields` cache implements in `gdb.Core` instead of `contrib/drivers`; add `ClearTableFields` and `ClearCache` functions for `gdb.Core` (#2128) * add ClearTableFiels/ClearCache for Core of package gdb * improve TableFields for contrib/drivers * fix UT case for contrib/drivers/clickhouse * remove unecessary attribute state for internal/rwmutex and internal/mutex * add ClearTableFieldsAll/ClearCacheAll for gdb.Core * improve clickhouse driver * improve clickhouse driver * fix ut * feat: improve import Co-authored-by: daguang <daguang830@gmail.com> Co-authored-by: houseme <housemecn@gmail.com> * refract builtin rules management mechanism, add `eq/not-eq/gt/gte/lt/lte/before/before-equal/after/after-equal/array/not-regex` rules for for package `gvalid` (#2133) * refract builtin rules management for package gvalid * refract builtin rules management for package gvalid * refract builtin rules management for package gvalid * add valiation rules and implements for package gvalid * UT cases update for package gvalid * improve error message of fields validation for package gvalid * up * add more validation rules for package gvalid * add validation rule foreach for package gvalid (#2136) * add ToSQL/CatchSQL funcions for package gdb (#2137) * add ToSQL/CatchSQL funcions for package gdb * Update gdb_core_underlying.go * fix ci Co-authored-by: houseme <housemecn@gmail.com> * add redis interface for package gredis * up * remove `FilteredLink` function for DB and all driver implements and improve details for package gdb (#2142) * fix: pgsql DoExec Transaction checks (#2101) Co-authored-by: John Guo <john@johng.cn> * improve package gdb * up * up * up * up * up * add DriverWrapper and DriverWarapperDB for package gdb * add DriverWrapper and DriverWarapperDB for package gdb * up Co-authored-by: HaiLaz <739476267@qq.com> * add new database driver `dm` * add drivers dm * upd go version * add gf ci yaml Co-authored-by: Xu <zhenghao.xu> * move go-redis implements from package gredis to contrib/nosql/redis; add redis string operation functions for package gredis * improve `contrib/drivers/dm` (#2144) * improve contrib/drivers/dm * format TODO list info * 1) add config.Name is required 2) The upper layer no longer needs to specify the schema 3) Adjust unit tests Co-authored-by: Xu <zhenghao.xu> Co-authored-by: houseme <housemecn@gmail.com> * move redis adapter related ut case from package gcache/gsession to package contrib/nosql/redis * up * up * up * up * up * improve comment * add implements of `gcfg.Adapter` using kubernetes configmap (#2145) * remove Logger from kubecm.Client * README updates for package kubecm * error message update for package gredis * comment update for package gdb * Feature/v2.2.0 gredis (#2155) * improve package gredis (#2162) * improve package gredis * Update gredis_redis_group_list.go * fix * up Co-authored-by: houseme <housemecn@gmail.com> * up * up * up * up * up * up * add func Test_GroupScript_Eval * ut cases for group string * UT cases update for group script * mv redis operation implements to contrib/nosql/redis from package gredis * test: add redis group list unit test (#2248) * test: add redis group list unit test * improve comment * test: fix redis group list unit test Co-authored-by: houseme <housemecn@gmail.com> * up * add func Test_GroupGeneric_Copy, Test_GroupGeneric_Exists,Test_GroupGeneric_Type,Test_GroupGeneric_Unlink,Test_GroupGeneric_Rename,Test_GroupGeneric_Move,Test_GroupGeneric_Del * add Redis GroupGeneric UnitTest (#2253) add func Test_GroupGeneric_RandomKey,Test_GroupGeneric_DBSize,Test_GroupGeneric_Keys,Test_GroupGeneric_FlushDB,Test_GroupGeneric_FlushAll,Test_GroupGeneric_Expire,Test_GroupGeneric_ExpireAt * hash test case completed (#2260) Co-authored-by: junler <sunjun@bookan.com> * add Redis GroupGeneric Unit Test part2 (#2258) * up * ci updates * ci updates * up * Feature/contrib redis fsprouts (#2274) * Feature/contrib redis starck (#2275) * up * up * fix `/*` router supported for handler of package ghttp; fix json tag name issue when it contains `,` for package goai; add proxy example for http server (#2294) * fix router supported for handler of package ghttp; fix json tag name issue when it contains for package goai * add proxy example for http server * fix: update szenius/set-timezone@v1.1 (#2293) * add Tag* functions to retreive most commonly used tag value from struct field for package gstructs; use description tag as default value if brief is empty for gcmd.Argument (#2299) * fix cache issue in Count/Value functions for gdb.Model (#2300) * add Tag* functions to retreive most commonly used tag value from struct field for package gstructs; use description tag as default value if brief is empty for gcmd.Argument * fix cache issue in Count/Value functions for gdb.Model * add more ut case for package gdb * version updates * add minus of `start` parameter support for `gstr.Substr`, like the `substr` function in `PHP` (#2297) * Make the substr like the substr in PHP Make the substr like the substr in PHP * Update gstr_z_unit_test.go * Update gstr_z_unit_test.go * Make the SubStrRune like the mb_substr in PHP Make the SubStrRune like the mb_substr in PHP * Update gstr_z_unit_test.go * Update gstr_z_unit_test.go * Update gins_z_unit_view_test.go * Update gview_z_unit_test.go * add ut cases for package gcode (#2307) * add ut cases for package gerror (#2304) * add ut cases for package gerror * add ut cases for package gerror * add ut cases for package gtime (#2303) * add ut cases for package gtime * add ut cases for package gtime * add ut cases for package gtime * add ut cases for package glog (#2302) * add ut cases for package glog * add ut cases for package glog * add ut cases for package glog * add ut cases for package glog * add ut cases for package glog * add ut cases for package glog * change result data type of function Count from int to int64 for package gdb (#2298) * feat: modify model count value int64 * fix * fix:modify int64 * fix * feat: cmd gf prebuild suport oracle (#2312) * add ut cases for package g (#2315) * add ut cases for package gdebug (#2313) * add ut cases for package gdebug * add ut cases for package gdebug * add ut cases for package gdebug Co-authored-by: houseme <housemecn@gmail.com> * add zookeeper registry support (#2284) * add ut cases for package glog part2 (#2317) * fix invalid UpdatedAt usage in soft deleting feature for package gdb (#2323) * fix issue in failed installing when there's shortcut between file paths for command install (#2326) * fix issue in failed installing when has shortcut between file paths for command install * version updates * template for command gf updates * improve lru clearing for package gcache (#2327) * add ut cases for package ghttp_middleware and ghttp_request (#2344) * add ut cases for package ghttp_middleware * add ut cases for package ghttp_request * add ut cases for package ghttp_request * add ut cases for package ghttp_response (#2352) * add ut cases for package ghttp_response * add ut cases for package ghttp_response * add ut cases for package ghttp_response * add ut cases for package ghttp_request (#2351) * add ut cases for package ghttp_middleware * add ut cases for package ghttp_request * add ut cases for package ghttp_request * add ut cases for package ghttp_request * add ut cases for package ghttp_request - form * add ut cases for package ghttp_request - query * add ut cases for package ghttp_request - request * add ut cases for package ghttp_request - router * add ut cases for package gcache (#2341) * gTcp Example Function: 1.NewConn 2.NewConnTLS 3.NewConnKeyCrt * gTcp Example Function: 1.Send * add example function ExampleConn_Recv and ExampleConn_RecvWithTimeout * add example function 1. ExampleConn_SendWithTimeout 2. ExampleConn_RecvLine 3. ExampleConn_RecvTill * add example function 1. ExampleConn_SendRecv 2. ExampleConn_SendRecvWithTimeout 3. ExampleConn_SetDeadline 4. ExampleConn_SetReceiveBufferWait * add gtcp test function 1. Test_Package_Option_HeadSize4 2. Test_Package_Option_Error * add gtcp example function 1. ExampleGetFreePorts 2. ExampleSend 3. ExampleSendRecv 4. ExampleSendWithTimeout 5. ExampleSendRecvWithTimeout 6. ExampleMustGetFreePort * add gtcp example function 1. ExampleSendPkg 2. ExampleSendRecvPkg 3. ExampleSendPkgWithTimeout 4. ExampleSendRecvPkgWithTimeout * add gtcp test function 1. Test_Pool_Send 2. Test_Pool_Recv 3. Test_Pool_RecvLine 4. Test_Pool_RecvTill 5. Test_Pool_RecvWithTimeout 6. Test_Pool_SendWithTimeout 7. Test_Pool_SendRecvWithTimeout * fix * add gtcp example function 1. ExampleGetServer 2. ExampleSetAddress 3. ExampleSetHandler 4. ExampleRun_NilHandle * exec CI * exec CI * exec CI * modify test server address * modify and exec CI * modify and exec CI * modify and exec CI * modify and exec CI * modify and exec CI * modify and exec CI * add example funcion ExampleConn_Recv_Once and fix * fix * add some error case in example function * add some error case in example function * 1.add example function ExampleNewServerKeyCrt 2.add function SendRecvPkgWithTimeout unit test * add function Test_Server_NewServerKeyCrt unit test * revert * add function Test_Package_Timeout, Test_Package_Option_HeadSize3, Test_Conn_RecvPkgError unit test * fix * add example function 1.ExampleClient_Clone 2.ExampleLoadKeyCrt * add example function 1.ExampleNewNetConnKeyCrt * fix * add example function 1.ExampleClient_DeleteBytes 2.ExampleClient_HeadBytes 3.ExampleClient_PatchBytes 4.ExampleClient_ConnectBytes 5.ExampleClient_OptionsBytes 6.ExampleClient_TraceBytes 7.ExampleClient_PutBytes * add example function 1.ExampleClient_Prefix 2.ExampleClient_Retry 3.ExampleClient_RedirectLimit * add example function 1.ExampleClient_SetBrowserMode 2.ExampleClient_SetHeader 3.ExampleClient_SetRedirectLimit * add example function 1.ExampleClient_SetTLSKeyCrt 2.ExampleClient_SetTLSConfig modify example funcion 1.ExampleClient_SetProxy 2.ExampleClient_Proxy * add example function 1.ExampleClient_PutContent 2.ExampleClient_DeleteContent 3.ExampleClient_HeadContent 4.ExampleClient_PatchContent 5.ExampleClient_ConnectContent 6.ExampleClient_OptionsContent 7.ExampleClient_TraceContent 8.ExampleClient_RequestContent * add example function 1.ExampleClient_RawRequest * add unit function 1.TestGetFreePorts 2.TestNewConn 3.TestNewConnTLS 4.TestNewConnKeyCrt 5.TestConn_SendWithTimeout * add unit function 1.TestConn_Send 2.TestConn_SendRecv 3.TestConn_SendRecvWithTimeout * modify * modify * add example function 1.TestConn_SetReceiveBufferWait 2.TestNewNetConnKeyCrt 3.TestSend * add example function 1.TestSendRecv 2.TestSendWithTimeout * add unit function 1.TestMustGetFreePort 2.TestSendRecvWithTimeout 3.TestSendPkg * add client recevied server's response content assert * modify * modify * add example function 1.TestSendRecvPkg 2.TestSendPkgWithTimeout 3.TestSendRecvPkgWithTimeout * add GetAddress() function add unit funciton 1.TestNewServer 2.TestGetServer 3.TestServer_SetAddress 4.TestServer_SetHandler 5.TestServer_Run * modify * modify * add unit funciton 1.TestLoadKeyCrt * modify * delete function fromHex * add gclient dump unit test * add example function 1.ExampleClient_Put 2.ExampleClient_Delete 3.ExampleClient_Head 4.ExampleClient_Patch 5.ExampleClient_Connect 6.ExampleClient_Options 7.ExampleClient_Trace * add example function 1.TestClient_DoRequest * add example function 1.ExampleClient_PutVar 2.ExampleClient_DeleteVar 3.ExampleClient_HeadVar 4.ExampleClient_PatchVar 5.ExampleClient_ConnectVar 6.ExampleClient_OptionsVar 7.ExampleClient_TraceVar * modify * modify * add CustomProvider function * modify * add unit funciton 1.Test_NewConn 2.Test_GetFreePorts * add unit funciton 1.Test_Server * garray_normal_any code converage * garray_normal_int code converage * garray_normal_str code converage * garray_sorted_any code converage * garray_sorted_int code converage * garray_sorted_str code converage * glist code converage * gmap, gmap_hash_any_any_map code converage * gmap_hash_int_any_map code converage * gmap_hash_int_any_map code converage * gmap_hash_int_int_map code converage * gmap_hash_int_str_map code converage * gmap_hash_str_any_map code converage * gmap_hash_str_int_map code converage * gmap_hash_str_str_map code converage * gmap_list_map code converage * gmap_list_map code converage * revert gf.yml * add gtest unit test function * add ut cases for package gcache * add ut cases for package gcache * add ut cases for package gcache * add ut cases for package gcache * add ut cases for package gcache * modify Co-authored-by: John Guo <john@johng.cn> * improve ut case for package internal/rwmutex (#2364) * fix issue when only one file was uploaded in batch receiver attribute (#2365) * fix fixed An error occurred when only one file was uploaded in batches and add unit testing(#2092) * fix issue uploading files for ghttp.Server Co-authored-by: yxh <yxh1103@qq.com> * fix issue #2334 when accessing static files with cache time (#2366) * Solve the problem of error when accessing static files with cache time. Error message: 2022-11-29 19:40:11.090 [ERRO] http: superfluous response.WriteHeader call from github.com/gogf/gf/v2/net/ghttp.(*ResponseWriter).Flush (ghttp_response_writer.go:58) Stack: Verification method: curl 'http://127.0.0.1:8000/' -H 'If-Modified-Since: Thu, 08 Dec 2022 03:13:55 GMT' --compressed * Solve the problem of error when accessing static files with cache time. Error message: 2022-11-29 19:40:11.090 [ERRO] http: superfluous response.WriteHeader call from github.com/gogf/gf/v2/net/ghttp.(*ResponseWriter).Flush (ghttp_response_writer.go:58) Stack: Verification method: curl 'http://127.0.0.1:8000/' -H 'If-Modified-Since: Thu, 08 Dec 2022 03:13:55 GMT' --compressed * Solve the problem of error when accessing static files with cache time. Error message: 2022-11-29 19:40:11.090 [ERRO] http: superfluous response.WriteHeader call from github.com/gogf/gf/v2/net/ghttp.(*ResponseWriter).Flush (ghttp_response_writer.go:58) Stack: Verification method: curl 'http://127.0.0.1:8000/' -H 'If-Modified-Since: Thu, 08 Dec 2022 03:13:55 GMT' --compressed * fix issue #2334 when accessing static files with cache time * up Co-authored-by: 曾洪亮 <hongliang.zeng@i-soft.com.cn> Co-authored-by: houseme <housemecn@gmail.com> * fix issue in cycle dumping for g.Dump (#2367) * fix issue in cycle dumping for g.Dump * up * up * up Co-authored-by: houseme <housemecn@gmail.com> * 由于 clickhouse 的 position的初始值为 1,导致gdb_core_utility.HasField 中对 fieldsArray 初始化出错 (#2346) * 由于 clickhouse 的 position的初始值为 1,导致gdb_core_utility.HasField 中对 fieldsArray 初始化出错 * 修复单元测试 * 修复单元测试 * 补充单元测试 * 增加CK防御性代码 Co-authored-by: longl <longlei@dealmap.cloud> Co-authored-by: houseme <housemecn@gmail.com> * fix: ghttp server static path config (#2335) Co-authored-by: daguang <daguang830@gmail.com> Co-authored-by: houseme <housemecn@gmail.com> Co-authored-by: ftl <1139556759@qq.com> Co-authored-by: HaiLaz <739476267@qq.com> Co-authored-by: zhonghuaxunGM <50815786+zhonghuaxunGM@users.noreply.github.com> Co-authored-by: huangqian <huangqian1985@qq.com> Co-authored-by: junler <827640651@qq.com> Co-authored-by: junler <sunjun@bookan.com> Co-authored-by: Starccck <28645972+starccck@users.noreply.github.com> Co-authored-by: Jinhongyu <30454170+cnjinhy@users.noreply.github.com> Co-authored-by: YuanXin Hu <huyuanxin1999@outlook.com> Co-authored-by: yxh <yxh1103@qq.com> Co-authored-by: 曾洪亮 <hongliang.zeng@i-soft.com.cn> Co-authored-by: long <48313408+qq375251855@users.noreply.github.com> Co-authored-by: longl <longlei@dealmap.cloud> Co-authored-by: houseme <housemecn@gmail.com> Co-authored-by: daguang <daguang830@gmail.com> Co-authored-by: ftl <1139556759@qq.com> Co-authored-by: HaiLaz <739476267@qq.com> Co-authored-by: zhonghuaxunGM <50815786+zhonghuaxunGM@users.noreply.github.com> Co-authored-by: huangqian <huangqian1985@qq.com> Co-authored-by: junler <827640651@qq.com> Co-authored-by: junler <sunjun@bookan.com> Co-authored-by: Starccck <28645972+starccck@users.noreply.github.com> Co-authored-by: Jinhongyu <30454170+cnjinhy@users.noreply.github.com> Co-authored-by: YuanXin Hu <huyuanxin1999@outlook.com> Co-authored-by: yxh <yxh1103@qq.com> Co-authored-by: 曾洪亮 <hongliang.zeng@i-soft.com.cn> Co-authored-by: long <48313408+qq375251855@users.noreply.github.com> Co-authored-by: longl <longlei@dealmap.cloud>
520 lines
17 KiB
Go
520 lines
17 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
|
|
package gdb
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"reflect"
|
|
|
|
"github.com/gogf/gf/v2/container/gtype"
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
"github.com/gogf/gf/v2/internal/reflection"
|
|
"github.com/gogf/gf/v2/text/gregex"
|
|
"github.com/gogf/gf/v2/util/gconv"
|
|
)
|
|
|
|
// TXCore is the struct for transaction management.
|
|
type TXCore struct {
|
|
db DB // db is the current gdb database manager.
|
|
tx *sql.Tx // tx is the raw and underlying transaction manager.
|
|
ctx context.Context // ctx is the context for this transaction only.
|
|
master *sql.DB // master is the raw and underlying database manager.
|
|
transactionId string // transactionId is a unique id generated by this object for this transaction.
|
|
transactionCount int // transactionCount marks the times that Begins.
|
|
isClosed bool // isClosed marks this transaction has already been committed or rolled back.
|
|
}
|
|
|
|
const (
|
|
transactionPointerPrefix = "transaction"
|
|
contextTransactionKeyPrefix = "TransactionObjectForGroup_"
|
|
transactionIdForLoggerCtx = "TransactionId"
|
|
)
|
|
|
|
var transactionIdGenerator = gtype.NewUint64()
|
|
|
|
// Begin starts and returns the transaction object.
|
|
// You should call Commit or Rollback functions of the transaction object
|
|
// if you no longer use the transaction. Commit or Rollback functions will also
|
|
// close the transaction automatically.
|
|
func (c *Core) Begin(ctx context.Context) (tx TX, err error) {
|
|
return c.doBeginCtx(ctx)
|
|
}
|
|
|
|
func (c *Core) doBeginCtx(ctx context.Context) (TX, error) {
|
|
master, err := c.db.Master()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var out DoCommitOutput
|
|
out, err = c.db.DoCommit(ctx, DoCommitInput{
|
|
Db: master,
|
|
Sql: "BEGIN",
|
|
Type: SqlTypeBegin,
|
|
IsTransaction: true,
|
|
})
|
|
return out.Tx, err
|
|
}
|
|
|
|
// Transaction wraps the transaction logic using function `f`.
|
|
// It rollbacks the transaction and returns the error from function `f` if
|
|
// it returns non-nil error. It commits the transaction and returns nil if
|
|
// function `f` returns nil.
|
|
//
|
|
// Note that, you should not Commit or Rollback the transaction in function `f`
|
|
// as it is automatically handled by this function.
|
|
func (c *Core) Transaction(ctx context.Context, f func(ctx context.Context, tx TX) error) (err error) {
|
|
if ctx == nil {
|
|
ctx = c.db.GetCtx()
|
|
}
|
|
ctx = c.InjectInternalCtxData(ctx)
|
|
// Check transaction object from context.
|
|
var tx TX
|
|
tx = TXFromCtx(ctx, c.db.GetGroup())
|
|
if tx != nil {
|
|
return tx.Transaction(ctx, f)
|
|
}
|
|
tx, err = c.doBeginCtx(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Inject transaction object into context.
|
|
tx = tx.Ctx(WithTX(tx.GetCtx(), tx))
|
|
defer func() {
|
|
if err == nil {
|
|
if exception := recover(); exception != nil {
|
|
if v, ok := exception.(error); ok && gerror.HasStack(v) {
|
|
err = v
|
|
} else {
|
|
err = gerror.Newf("%+v", exception)
|
|
}
|
|
}
|
|
}
|
|
if err != nil {
|
|
if e := tx.Rollback(); e != nil {
|
|
err = e
|
|
}
|
|
} else {
|
|
if e := tx.Commit(); e != nil {
|
|
err = e
|
|
}
|
|
}
|
|
}()
|
|
err = f(tx.GetCtx(), tx)
|
|
return
|
|
}
|
|
|
|
// WithTX injects given transaction object into context and returns a new context.
|
|
func WithTX(ctx context.Context, tx TX) context.Context {
|
|
if tx == nil {
|
|
return ctx
|
|
}
|
|
// Check repeat injection from given.
|
|
group := tx.GetDB().GetGroup()
|
|
if ctxTx := TXFromCtx(ctx, group); ctxTx != nil && ctxTx.GetDB().GetGroup() == group {
|
|
return ctx
|
|
}
|
|
dbCtx := tx.GetDB().GetCtx()
|
|
if ctxTx := TXFromCtx(dbCtx, group); ctxTx != nil && ctxTx.GetDB().GetGroup() == group {
|
|
return dbCtx
|
|
}
|
|
// Inject transaction object and id into context.
|
|
ctx = context.WithValue(ctx, transactionKeyForContext(group), tx)
|
|
return ctx
|
|
}
|
|
|
|
// TXFromCtx retrieves and returns transaction object from context.
|
|
// It is usually used in nested transaction feature, and it returns nil if it is not set previously.
|
|
func TXFromCtx(ctx context.Context, group string) TX {
|
|
if ctx == nil {
|
|
return nil
|
|
}
|
|
v := ctx.Value(transactionKeyForContext(group))
|
|
if v != nil {
|
|
tx := v.(TX)
|
|
if tx.IsClosed() {
|
|
return nil
|
|
}
|
|
tx = tx.Ctx(ctx)
|
|
return tx
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// transactionKeyForContext forms and returns a string for storing transaction object of certain database group into context.
|
|
func transactionKeyForContext(group string) string {
|
|
return contextTransactionKeyPrefix + group
|
|
}
|
|
|
|
// transactionKeyForNestedPoint forms and returns the transaction key at current save point.
|
|
func (tx *TXCore) transactionKeyForNestedPoint() string {
|
|
return tx.db.GetCore().QuoteWord(transactionPointerPrefix + gconv.String(tx.transactionCount))
|
|
}
|
|
|
|
// Ctx sets the context for current transaction.
|
|
func (tx *TXCore) Ctx(ctx context.Context) TX {
|
|
tx.ctx = ctx
|
|
if tx.ctx != nil {
|
|
tx.ctx = tx.db.GetCore().InjectInternalCtxData(tx.ctx)
|
|
}
|
|
return tx
|
|
}
|
|
|
|
// GetCtx returns the context for current transaction.
|
|
func (tx *TXCore) GetCtx() context.Context {
|
|
return tx.ctx
|
|
}
|
|
|
|
// GetDB returns the DB for current transaction.
|
|
func (tx *TXCore) GetDB() DB {
|
|
return tx.db
|
|
}
|
|
|
|
// GetSqlTX returns the underlying transaction object for current transaction.
|
|
func (tx *TXCore) GetSqlTX() *sql.Tx {
|
|
return tx.tx
|
|
}
|
|
|
|
// Commit commits current transaction.
|
|
// Note that it releases previous saved transaction point if it's in a nested transaction procedure,
|
|
// or else it commits the hole transaction.
|
|
func (tx *TXCore) Commit() error {
|
|
if tx.transactionCount > 0 {
|
|
tx.transactionCount--
|
|
_, err := tx.Exec("RELEASE SAVEPOINT " + tx.transactionKeyForNestedPoint())
|
|
return err
|
|
}
|
|
_, err := tx.db.DoCommit(tx.ctx, DoCommitInput{
|
|
Tx: tx.tx,
|
|
Sql: "COMMIT",
|
|
Type: SqlTypeTXCommit,
|
|
IsTransaction: true,
|
|
})
|
|
if err == nil {
|
|
tx.isClosed = true
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Rollback aborts current transaction.
|
|
// Note that it aborts current transaction if it's in a nested transaction procedure,
|
|
// or else it aborts the hole transaction.
|
|
func (tx *TXCore) Rollback() error {
|
|
if tx.transactionCount > 0 {
|
|
tx.transactionCount--
|
|
_, err := tx.Exec("ROLLBACK TO SAVEPOINT " + tx.transactionKeyForNestedPoint())
|
|
return err
|
|
}
|
|
_, err := tx.db.DoCommit(tx.ctx, DoCommitInput{
|
|
Tx: tx.tx,
|
|
Sql: "ROLLBACK",
|
|
Type: SqlTypeTXRollback,
|
|
IsTransaction: true,
|
|
})
|
|
if err == nil {
|
|
tx.isClosed = true
|
|
}
|
|
return err
|
|
}
|
|
|
|
// IsClosed checks and returns this transaction has already been committed or rolled back.
|
|
func (tx *TXCore) IsClosed() bool {
|
|
return tx.isClosed
|
|
}
|
|
|
|
// Begin starts a nested transaction procedure.
|
|
func (tx *TXCore) Begin() error {
|
|
_, err := tx.Exec("SAVEPOINT " + tx.transactionKeyForNestedPoint())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
tx.transactionCount++
|
|
return nil
|
|
}
|
|
|
|
// SavePoint performs `SAVEPOINT xxx` SQL statement that saves transaction at current point.
|
|
// The parameter `point` specifies the point name that will be saved to server.
|
|
func (tx *TXCore) SavePoint(point string) error {
|
|
_, err := tx.Exec("SAVEPOINT " + tx.db.GetCore().QuoteWord(point))
|
|
return err
|
|
}
|
|
|
|
// RollbackTo performs `ROLLBACK TO SAVEPOINT xxx` SQL statement that rollbacks to specified saved transaction.
|
|
// The parameter `point` specifies the point name that was saved previously.
|
|
func (tx *TXCore) RollbackTo(point string) error {
|
|
_, err := tx.Exec("ROLLBACK TO SAVEPOINT " + tx.db.GetCore().QuoteWord(point))
|
|
return err
|
|
}
|
|
|
|
// Transaction wraps the transaction logic using function `f`.
|
|
// It rollbacks the transaction and returns the error from function `f` if
|
|
// it returns non-nil error. It commits the transaction and returns nil if
|
|
// function `f` returns nil.
|
|
//
|
|
// Note that, you should not Commit or Rollback the transaction in function `f`
|
|
// as it is automatically handled by this function.
|
|
func (tx *TXCore) Transaction(ctx context.Context, f func(ctx context.Context, tx TX) error) (err error) {
|
|
if ctx != nil {
|
|
tx.ctx = ctx
|
|
}
|
|
// Check transaction object from context.
|
|
if TXFromCtx(tx.ctx, tx.db.GetGroup()) == nil {
|
|
// Inject transaction object into context.
|
|
tx.ctx = WithTX(tx.ctx, tx)
|
|
}
|
|
err = tx.Begin()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if err == nil {
|
|
if exception := recover(); exception != nil {
|
|
if v, ok := exception.(error); ok && gerror.HasStack(v) {
|
|
err = v
|
|
} else {
|
|
err = gerror.Newf("%+v", exception)
|
|
}
|
|
}
|
|
}
|
|
if err != nil {
|
|
if e := tx.Rollback(); e != nil {
|
|
err = e
|
|
}
|
|
} else {
|
|
if e := tx.Commit(); e != nil {
|
|
err = e
|
|
}
|
|
}
|
|
}()
|
|
err = f(tx.ctx, tx)
|
|
return
|
|
}
|
|
|
|
// Query does query operation on transaction.
|
|
// See Core.Query.
|
|
func (tx *TXCore) Query(sql string, args ...interface{}) (result Result, err error) {
|
|
return tx.db.DoQuery(tx.ctx, &txLink{tx.tx}, sql, args...)
|
|
}
|
|
|
|
// Exec does none query operation on transaction.
|
|
// See Core.Exec.
|
|
func (tx *TXCore) Exec(sql string, args ...interface{}) (sql.Result, error) {
|
|
return tx.db.DoExec(tx.ctx, &txLink{tx.tx}, sql, args...)
|
|
}
|
|
|
|
// Prepare creates a prepared statement for later queries or executions.
|
|
// Multiple queries or executions may be run concurrently from the
|
|
// returned statement.
|
|
// The caller must call the statement's Close method
|
|
// when the statement is no longer needed.
|
|
func (tx *TXCore) Prepare(sql string) (*Stmt, error) {
|
|
return tx.db.DoPrepare(tx.ctx, &txLink{tx.tx}, sql)
|
|
}
|
|
|
|
// GetAll queries and returns data records from database.
|
|
func (tx *TXCore) GetAll(sql string, args ...interface{}) (Result, error) {
|
|
return tx.Query(sql, args...)
|
|
}
|
|
|
|
// GetOne queries and returns one record from database.
|
|
func (tx *TXCore) GetOne(sql string, args ...interface{}) (Record, error) {
|
|
list, err := tx.GetAll(sql, args...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(list) > 0 {
|
|
return list[0], nil
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
// GetStruct queries one record from database and converts it to given struct.
|
|
// The parameter `pointer` should be a pointer to struct.
|
|
func (tx *TXCore) GetStruct(obj interface{}, sql string, args ...interface{}) error {
|
|
one, err := tx.GetOne(sql, args...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return one.Struct(obj)
|
|
}
|
|
|
|
// GetStructs queries records from database and converts them to given struct.
|
|
// The parameter `pointer` should be type of struct slice: []struct/[]*struct.
|
|
func (tx *TXCore) GetStructs(objPointerSlice interface{}, sql string, args ...interface{}) error {
|
|
all, err := tx.GetAll(sql, args...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return all.Structs(objPointerSlice)
|
|
}
|
|
|
|
// GetScan queries one or more records from database and converts them to given struct or
|
|
// struct array.
|
|
//
|
|
// If parameter `pointer` is type of struct pointer, it calls GetStruct internally for
|
|
// the conversion. If parameter `pointer` is type of slice, it calls GetStructs internally
|
|
// for conversion.
|
|
func (tx *TXCore) GetScan(pointer interface{}, sql string, args ...interface{}) error {
|
|
reflectInfo := reflection.OriginTypeAndKind(pointer)
|
|
if reflectInfo.InputKind != reflect.Ptr {
|
|
return gerror.NewCodef(
|
|
gcode.CodeInvalidParameter,
|
|
"params should be type of pointer, but got: %v",
|
|
reflectInfo.InputKind,
|
|
)
|
|
}
|
|
switch reflectInfo.OriginKind {
|
|
case reflect.Array, reflect.Slice:
|
|
return tx.GetStructs(pointer, sql, args...)
|
|
|
|
case reflect.Struct:
|
|
return tx.GetStruct(pointer, sql, args...)
|
|
}
|
|
return gerror.NewCodef(
|
|
gcode.CodeInvalidParameter,
|
|
`in valid parameter type "%v", of which element type should be type of struct/slice`,
|
|
reflectInfo.InputType,
|
|
)
|
|
}
|
|
|
|
// GetValue queries and returns the field value from database.
|
|
// The sql should query only one field from database, or else it returns only one
|
|
// field of the result.
|
|
func (tx *TXCore) GetValue(sql string, args ...interface{}) (Value, error) {
|
|
one, err := tx.GetOne(sql, args...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, v := range one {
|
|
return v, nil
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
// GetCount queries and returns the count from database.
|
|
func (tx *TXCore) GetCount(sql string, args ...interface{}) (int64, error) {
|
|
if !gregex.IsMatchString(`(?i)SELECT\s+COUNT\(.+\)\s+FROM`, sql) {
|
|
sql, _ = gregex.ReplaceString(`(?i)(SELECT)\s+(.+)\s+(FROM)`, `$1 COUNT($2) $3`, sql)
|
|
}
|
|
value, err := tx.GetValue(sql, args...)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return value.Int64(), nil
|
|
}
|
|
|
|
// Insert does "INSERT INTO ..." statement for the table.
|
|
// If there's already one unique record of the data in the table, it returns error.
|
|
//
|
|
// The parameter `data` can be type of map/gmap/struct/*struct/[]map/[]struct, etc.
|
|
// Eg:
|
|
// Data(g.Map{"uid": 10000, "name":"john"})
|
|
// Data(g.Slice{g.Map{"uid": 10000, "name":"john"}, g.Map{"uid": 20000, "name":"smith"})
|
|
//
|
|
// The parameter `batch` specifies the batch operation count when given data is slice.
|
|
func (tx *TXCore) Insert(table string, data interface{}, batch ...int) (sql.Result, error) {
|
|
if len(batch) > 0 {
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Batch(batch[0]).Insert()
|
|
}
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Insert()
|
|
}
|
|
|
|
// InsertIgnore does "INSERT IGNORE INTO ..." statement for the table.
|
|
// If there's already one unique record of the data in the table, it ignores the inserting.
|
|
//
|
|
// The parameter `data` can be type of map/gmap/struct/*struct/[]map/[]struct, etc.
|
|
// Eg:
|
|
// Data(g.Map{"uid": 10000, "name":"john"})
|
|
// Data(g.Slice{g.Map{"uid": 10000, "name":"john"}, g.Map{"uid": 20000, "name":"smith"})
|
|
//
|
|
// The parameter `batch` specifies the batch operation count when given data is slice.
|
|
func (tx *TXCore) InsertIgnore(table string, data interface{}, batch ...int) (sql.Result, error) {
|
|
if len(batch) > 0 {
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Batch(batch[0]).InsertIgnore()
|
|
}
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).InsertIgnore()
|
|
}
|
|
|
|
// InsertAndGetId performs action Insert and returns the last insert id that automatically generated.
|
|
func (tx *TXCore) InsertAndGetId(table string, data interface{}, batch ...int) (int64, error) {
|
|
if len(batch) > 0 {
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Batch(batch[0]).InsertAndGetId()
|
|
}
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).InsertAndGetId()
|
|
}
|
|
|
|
// Replace does "REPLACE INTO ..." statement for the table.
|
|
// If there's already one unique record of the data in the table, it deletes the record
|
|
// and inserts a new one.
|
|
//
|
|
// The parameter `data` can be type of map/gmap/struct/*struct/[]map/[]struct, etc.
|
|
// Eg:
|
|
// Data(g.Map{"uid": 10000, "name":"john"})
|
|
// Data(g.Slice{g.Map{"uid": 10000, "name":"john"}, g.Map{"uid": 20000, "name":"smith"})
|
|
//
|
|
// The parameter `data` can be type of map/gmap/struct/*struct/[]map/[]struct, etc.
|
|
// If given data is type of slice, it then does batch replacing, and the optional parameter
|
|
// `batch` specifies the batch operation count.
|
|
func (tx *TXCore) Replace(table string, data interface{}, batch ...int) (sql.Result, error) {
|
|
if len(batch) > 0 {
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Batch(batch[0]).Replace()
|
|
}
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Replace()
|
|
}
|
|
|
|
// Save does "INSERT INTO ... ON DUPLICATE KEY UPDATE..." statement for the table.
|
|
// It updates the record if there's primary or unique index in the saving data,
|
|
// or else it inserts a new record into the table.
|
|
//
|
|
// The parameter `data` can be type of map/gmap/struct/*struct/[]map/[]struct, etc.
|
|
// Eg:
|
|
// Data(g.Map{"uid": 10000, "name":"john"})
|
|
// Data(g.Slice{g.Map{"uid": 10000, "name":"john"}, g.Map{"uid": 20000, "name":"smith"})
|
|
//
|
|
// If given data is type of slice, it then does batch saving, and the optional parameter
|
|
// `batch` specifies the batch operation count.
|
|
func (tx *TXCore) Save(table string, data interface{}, batch ...int) (sql.Result, error) {
|
|
if len(batch) > 0 {
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Batch(batch[0]).Save()
|
|
}
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Save()
|
|
}
|
|
|
|
// Update does "UPDATE ... " statement for the table.
|
|
//
|
|
// The parameter `data` can be type of string/map/gmap/struct/*struct, etc.
|
|
// Eg: "uid=10000", "uid", 10000, g.Map{"uid": 10000, "name":"john"}
|
|
//
|
|
// The parameter `condition` can be type of string/map/gmap/slice/struct/*struct, etc.
|
|
// It is commonly used with parameter `args`.
|
|
// Eg:
|
|
// "uid=10000",
|
|
// "uid", 10000
|
|
// "money>? AND name like ?", 99999, "vip_%"
|
|
// "status IN (?)", g.Slice{1,2,3}
|
|
// "age IN(?,?)", 18, 50
|
|
// User{ Id : 1, UserName : "john"}.
|
|
func (tx *TXCore) Update(table string, data interface{}, condition interface{}, args ...interface{}) (sql.Result, error) {
|
|
return tx.Model(table).Ctx(tx.ctx).Data(data).Where(condition, args...).Update()
|
|
}
|
|
|
|
// Delete does "DELETE FROM ... " statement for the table.
|
|
//
|
|
// The parameter `condition` can be type of string/map/gmap/slice/struct/*struct, etc.
|
|
// It is commonly used with parameter `args`.
|
|
// Eg:
|
|
// "uid=10000",
|
|
// "uid", 10000
|
|
// "money>? AND name like ?", 99999, "vip_%"
|
|
// "status IN (?)", g.Slice{1,2,3}
|
|
// "age IN(?,?)", 18, 50
|
|
// User{ Id : 1, UserName : "john"}.
|
|
func (tx *TXCore) Delete(table string, condition interface{}, args ...interface{}) (sql.Result, error) {
|
|
return tx.Model(table).Ctx(tx.ctx).Where(condition, args...).Delete()
|
|
}
|