mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 11:18:50 +08:00
* 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>
451 lines
14 KiB
Go
451 lines
14 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 clickhouse implements gdb.Driver, which supports operations for database ClickHouse.
|
|
package clickhouse
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"database/sql/driver"
|
|
"errors"
|
|
"fmt"
|
|
"net/url"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/ClickHouse/clickhouse-go/v2"
|
|
"github.com/gogf/gf/v2/database/gdb"
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
"github.com/gogf/gf/v2/os/gctx"
|
|
"github.com/gogf/gf/v2/os/gtime"
|
|
"github.com/gogf/gf/v2/text/gregex"
|
|
"github.com/gogf/gf/v2/util/gconv"
|
|
"github.com/gogf/gf/v2/util/gtag"
|
|
"github.com/gogf/gf/v2/util/gutil"
|
|
"github.com/google/uuid"
|
|
"github.com/shopspring/decimal"
|
|
)
|
|
|
|
// Driver is the driver for postgresql database.
|
|
type Driver struct {
|
|
*gdb.Core
|
|
}
|
|
|
|
var (
|
|
errUnsupportedInsertIgnore = errors.New("unsupported method:InsertIgnore")
|
|
errUnsupportedInsertGetId = errors.New("unsupported method:InsertGetId")
|
|
errUnsupportedReplace = errors.New("unsupported method:Replace")
|
|
errUnsupportedBegin = errors.New("unsupported method:Begin")
|
|
errUnsupportedTransaction = errors.New("unsupported method:Transaction")
|
|
)
|
|
|
|
const (
|
|
updateFilterPattern = `(?i)UPDATE[\s]+?(\w+[\.]?\w+)[\s]+?SET`
|
|
deleteFilterPattern = `(?i)DELETE[\s]+?FROM[\s]+?(\w+[\.]?\w+)`
|
|
filterTypePattern = `(?i)^UPDATE|DELETE`
|
|
replaceSchemaPattern = `@(.+?)/([\w\.\-]+)+`
|
|
needParsedSqlInCtx gctx.StrKey = "NeedParsedSql"
|
|
OrmTagForStruct = gtag.ORM
|
|
driverName = "clickhouse"
|
|
)
|
|
|
|
func init() {
|
|
if err := gdb.Register(`clickhouse`, New()); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
// New create and returns a driver that implements gdb.Driver, which supports operations for clickhouse.
|
|
func New() gdb.Driver {
|
|
return &Driver{}
|
|
}
|
|
|
|
// New creates and returns a database object for clickhouse.
|
|
// It implements the interface of gdb.Driver for extra database driver installation.
|
|
func (d *Driver) New(core *gdb.Core, node *gdb.ConfigNode) (gdb.DB, error) {
|
|
return &Driver{
|
|
Core: core,
|
|
}, nil
|
|
}
|
|
|
|
// Open creates and returns an underlying sql.DB object for clickhouse.
|
|
func (d *Driver) Open(config *gdb.ConfigNode) (db *sql.DB, err error) {
|
|
source := config.Link
|
|
// clickhouse://username:password@host1:9000,host2:9000/database?dial_timeout=200ms&max_execution_time=60
|
|
if config.Link != "" {
|
|
// ============================================================================
|
|
// Deprecated from v2.2.0.
|
|
// ============================================================================
|
|
// Custom changing the schema in runtime.
|
|
if config.Name != "" {
|
|
source, _ = gregex.ReplaceString(replaceSchemaPattern, "@$1/"+config.Name, config.Link)
|
|
} else {
|
|
// If no schema, the link is matched for replacement
|
|
dbName, _ := gregex.MatchString(replaceSchemaPattern, config.Link)
|
|
if len(dbName) > 0 {
|
|
config.Name = dbName[len(dbName)-1]
|
|
}
|
|
}
|
|
} else {
|
|
if config.Pass != "" {
|
|
source = fmt.Sprintf(
|
|
"clickhouse://%s:%s@%s:%s/%s?debug=%t",
|
|
config.User, url.PathEscape(config.Pass),
|
|
config.Host, config.Port, config.Name, config.Debug,
|
|
)
|
|
} else {
|
|
source = fmt.Sprintf(
|
|
"clickhouse://%s@%s:%s/%s?debug=%t",
|
|
config.User, config.Host, config.Port, config.Name, config.Debug,
|
|
)
|
|
}
|
|
if config.Extra != "" {
|
|
source = fmt.Sprintf("%s&%s", source, config.Extra)
|
|
}
|
|
}
|
|
if db, err = sql.Open(driverName, source); err != nil {
|
|
err = gerror.WrapCodef(
|
|
gcode.CodeDbOperationError, err,
|
|
`sql.Open failed for driver "%s" by source "%s"`, driverName, source,
|
|
)
|
|
return nil, err
|
|
}
|
|
return
|
|
}
|
|
|
|
// Tables retrieves and returns the tables of current schema.
|
|
// It's mainly used in cli tool chain for automatically generating the models.
|
|
func (d *Driver) Tables(ctx context.Context, schema ...string) (tables []string, err error) {
|
|
var result gdb.Result
|
|
link, err := d.SlaveLink(schema...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
query := fmt.Sprintf("select name from `system`.tables where database = '%s'", d.GetConfig().Name)
|
|
result, err = d.DoSelect(ctx, link, query)
|
|
if err != nil {
|
|
return
|
|
}
|
|
for _, m := range result {
|
|
tables = append(tables, m["name"].String())
|
|
}
|
|
return
|
|
}
|
|
|
|
// TableFields retrieves and returns the fields' information of specified table of current schema.
|
|
// Also see DriverMysql.TableFields.
|
|
func (d *Driver) TableFields(
|
|
ctx context.Context, table string, schema ...string,
|
|
) (fields map[string]*gdb.TableField, err error) {
|
|
var (
|
|
result gdb.Result
|
|
link gdb.Link
|
|
useSchema = gutil.GetOrDefaultStr(d.GetSchema(), schema...)
|
|
)
|
|
if link, err = d.SlaveLink(useSchema); err != nil {
|
|
return nil, err
|
|
}
|
|
var (
|
|
columns = "name,position,default_expression,comment,type,is_in_partition_key,is_in_sorting_key,is_in_primary_key,is_in_sampling_key"
|
|
getColumnsSql = fmt.Sprintf(
|
|
"select %s from `system`.columns c where `table` = '%s'",
|
|
columns, table,
|
|
)
|
|
)
|
|
result, err = d.DoSelect(ctx, link, getColumnsSql)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
fields = make(map[string]*gdb.TableField)
|
|
for _, m := range result {
|
|
var (
|
|
isNull = false
|
|
fieldType = m["type"].String()
|
|
)
|
|
// in clickhouse , filed type like is Nullable(int)
|
|
fieldsResult, _ := gregex.MatchString(`^Nullable\((.*?)\)`, fieldType)
|
|
if len(fieldsResult) == 2 {
|
|
isNull = true
|
|
fieldType = fieldsResult[1]
|
|
}
|
|
position := m["position"].Int()
|
|
if result[0]["position"].Int() != 0 {
|
|
position -= 1
|
|
}
|
|
fields[m["name"].String()] = &gdb.TableField{
|
|
Index: position,
|
|
Name: m["name"].String(),
|
|
Default: m["default_expression"].Val(),
|
|
Comment: m["comment"].String(),
|
|
// Key: m["Key"].String(),
|
|
Type: fieldType,
|
|
Null: isNull,
|
|
}
|
|
}
|
|
return fields, nil
|
|
}
|
|
|
|
// PingMaster pings the master node to check authentication or keeps the connection alive.
|
|
func (d *Driver) PingMaster() error {
|
|
conn, err := d.Master()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return d.ping(conn)
|
|
}
|
|
|
|
// PingSlave pings the slave node to check authentication or keeps the connection alive.
|
|
func (d *Driver) PingSlave() error {
|
|
conn, err := d.Slave()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return d.ping(conn)
|
|
}
|
|
|
|
// ping Returns the Clickhouse specific error.
|
|
func (d *Driver) ping(conn *sql.DB) error {
|
|
err := conn.Ping()
|
|
if exception, ok := err.(*clickhouse.Exception); ok {
|
|
return fmt.Errorf("[%d]%s", exception.Code, exception.Message)
|
|
}
|
|
return err
|
|
}
|
|
|
|
// DoFilter handles the sql before posts it to database.
|
|
func (d *Driver) DoFilter(
|
|
ctx context.Context, link gdb.Link, originSql string, args []interface{},
|
|
) (newSql string, newArgs []interface{}, err error) {
|
|
if len(args) == 0 {
|
|
return originSql, args, nil
|
|
}
|
|
|
|
var index int
|
|
// Convert placeholder char '?' to string "$x".
|
|
originSql, _ = gregex.ReplaceStringFunc(`\?`, originSql, func(s string) string {
|
|
index++
|
|
return fmt.Sprintf(`$%d`, index)
|
|
})
|
|
|
|
// Only SQL generated through the framework is processed.
|
|
if !d.getNeedParsedSqlFromCtx(ctx) {
|
|
return originSql, args, nil
|
|
}
|
|
|
|
// replace STD SQL to Clickhouse SQL grammar
|
|
modeRes, err := gregex.MatchString(filterTypePattern, strings.TrimSpace(originSql))
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
if len(modeRes) == 0 {
|
|
return originSql, args, nil
|
|
}
|
|
|
|
// Only delete/ UPDATE statements require filter
|
|
switch strings.ToUpper(modeRes[0]) {
|
|
case "UPDATE":
|
|
// MySQL eg: UPDATE table_name SET field1=new-value1, field2=new-value2 [WHERE Clause]
|
|
// Clickhouse eg: ALTER TABLE [db.]table UPDATE column1 = expr1 [, ...] WHERE filter_expr
|
|
newSql, err = gregex.ReplaceStringFuncMatch(updateFilterPattern, originSql, func(s []string) string {
|
|
return fmt.Sprintf("ALTER TABLE %s UPDATE", s[1])
|
|
})
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
return newSql, args, nil
|
|
|
|
case "DELETE":
|
|
// MySQL eg: DELETE FROM table_name [WHERE Clause]
|
|
// Clickhouse eg: ALTER TABLE [db.]table [ON CLUSTER cluster] DELETE WHERE filter_expr
|
|
newSql, err = gregex.ReplaceStringFuncMatch(deleteFilterPattern, originSql, func(s []string) string {
|
|
return fmt.Sprintf("ALTER TABLE %s DELETE", s[1])
|
|
})
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
return newSql, args, nil
|
|
|
|
}
|
|
return originSql, args, nil
|
|
}
|
|
|
|
// DoCommit commits current sql and arguments to underlying sql driver.
|
|
func (d *Driver) DoCommit(ctx context.Context, in gdb.DoCommitInput) (out gdb.DoCommitOutput, err error) {
|
|
ctx = d.InjectIgnoreResult(ctx)
|
|
return d.Core.DoCommit(ctx, in)
|
|
}
|
|
|
|
func (d *Driver) DoInsert(
|
|
ctx context.Context, link gdb.Link, table string, list gdb.List, option gdb.DoInsertOption,
|
|
) (result sql.Result, err error) {
|
|
var (
|
|
keys []string // Field names.
|
|
valueHolder = make([]string, 0)
|
|
)
|
|
// Handle the field names and placeholders.
|
|
for k := range list[0] {
|
|
keys = append(keys, k)
|
|
valueHolder = append(valueHolder, "?")
|
|
}
|
|
// Prepare the batch result pointer.
|
|
var (
|
|
charL, charR = d.Core.GetChars()
|
|
keysStr = charL + strings.Join(keys, charR+","+charL) + charR
|
|
holderStr = strings.Join(valueHolder, ",")
|
|
tx gdb.TX
|
|
stdSqlResult sql.Result
|
|
stmt *gdb.Stmt
|
|
)
|
|
tx, err = d.Core.Begin(ctx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
stmt, err = tx.Prepare(fmt.Sprintf(
|
|
"INSERT INTO %s(%s) VALUES (%s)",
|
|
d.QuotePrefixTableName(table), keysStr,
|
|
holderStr,
|
|
))
|
|
if err != nil {
|
|
return
|
|
}
|
|
for i := 0; i < len(list); i++ {
|
|
params := make([]interface{}, 0) // Values that will be committed to underlying database driver.
|
|
for _, k := range keys {
|
|
params = append(params, list[i][k])
|
|
}
|
|
// Prepare is allowed to execute only once in a transaction opened by clickhouse
|
|
stdSqlResult, err = stmt.ExecContext(ctx, params...)
|
|
if err != nil {
|
|
return stdSqlResult, err
|
|
}
|
|
}
|
|
return stdSqlResult, tx.Commit()
|
|
}
|
|
|
|
// ConvertDataForRecord converting for any data that will be inserted into table/collection as a record.
|
|
func (d *Driver) ConvertDataForRecord(ctx context.Context, value interface{}) (map[string]interface{}, error) {
|
|
m := gconv.Map(value, OrmTagForStruct)
|
|
|
|
// transforms a value of a particular type
|
|
for k, v := range m {
|
|
switch itemValue := v.(type) {
|
|
|
|
case time.Time:
|
|
m[k] = itemValue
|
|
// If the time is zero, it then updates it to nil,
|
|
// which will insert/update the value to database as "null".
|
|
if itemValue.IsZero() {
|
|
m[k] = nil
|
|
}
|
|
|
|
case uuid.UUID:
|
|
m[k] = itemValue
|
|
|
|
case *time.Time:
|
|
m[k] = itemValue
|
|
// If the time is zero, it then updates it to nil,
|
|
// which will insert/update the value to database as "null".
|
|
if itemValue == nil || itemValue.IsZero() {
|
|
m[k] = nil
|
|
}
|
|
|
|
case gtime.Time:
|
|
// for gtime type, needs to get time.Time
|
|
m[k] = itemValue.Time
|
|
// If the time is zero, it then updates it to nil,
|
|
// which will insert/update the value to database as "null".
|
|
if itemValue.IsZero() {
|
|
m[k] = nil
|
|
}
|
|
|
|
case *gtime.Time:
|
|
// for gtime type, needs to get time.Time
|
|
if itemValue != nil {
|
|
m[k] = itemValue.Time
|
|
}
|
|
// If the time is zero, it then updates it to nil,
|
|
// which will insert/update the value to database as "null".
|
|
if itemValue == nil || itemValue.IsZero() {
|
|
m[k] = nil
|
|
}
|
|
|
|
case decimal.Decimal:
|
|
m[k] = itemValue
|
|
|
|
case *decimal.Decimal:
|
|
m[k] = nil
|
|
if itemValue != nil {
|
|
m[k] = *itemValue
|
|
}
|
|
|
|
default:
|
|
// if the other type implements valuer for the driver package
|
|
// the converted result is used
|
|
// otherwise the interface data is committed
|
|
valuer, ok := itemValue.(driver.Valuer)
|
|
if !ok {
|
|
m[k] = itemValue
|
|
continue
|
|
}
|
|
convertedValue, err := valuer.Value()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m[k] = convertedValue
|
|
}
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func (d *Driver) DoDelete(ctx context.Context, link gdb.Link, table string, condition string, args ...interface{}) (result sql.Result, err error) {
|
|
ctx = d.injectNeedParsedSql(ctx)
|
|
return d.Core.DoDelete(ctx, link, table, condition, args...)
|
|
}
|
|
|
|
func (d *Driver) DoUpdate(ctx context.Context, link gdb.Link, table string, data interface{}, condition string, args ...interface{}) (result sql.Result, err error) {
|
|
ctx = d.injectNeedParsedSql(ctx)
|
|
return d.Core.DoUpdate(ctx, link, table, data, condition, args...)
|
|
}
|
|
|
|
// InsertIgnore Other queries for modifying data parts are not supported: REPLACE, MERGE, UPSERT, INSERT UPDATE.
|
|
func (d *Driver) InsertIgnore(ctx context.Context, table string, data interface{}, batch ...int) (sql.Result, error) {
|
|
return nil, errUnsupportedInsertIgnore
|
|
}
|
|
|
|
// InsertAndGetId Other queries for modifying data parts are not supported: REPLACE, MERGE, UPSERT, INSERT UPDATE.
|
|
func (d *Driver) InsertAndGetId(ctx context.Context, table string, data interface{}, batch ...int) (int64, error) {
|
|
return 0, errUnsupportedInsertGetId
|
|
}
|
|
|
|
// Replace Other queries for modifying data parts are not supported: REPLACE, MERGE, UPSERT, INSERT UPDATE.
|
|
func (d *Driver) Replace(ctx context.Context, table string, data interface{}, batch ...int) (sql.Result, error) {
|
|
return nil, errUnsupportedReplace
|
|
}
|
|
|
|
func (d *Driver) Begin(ctx context.Context) (tx gdb.TX, err error) {
|
|
return nil, errUnsupportedBegin
|
|
}
|
|
|
|
func (d *Driver) Transaction(ctx context.Context, f func(ctx context.Context, tx gdb.TX) error) error {
|
|
return errUnsupportedTransaction
|
|
}
|
|
|
|
func (d *Driver) injectNeedParsedSql(ctx context.Context) context.Context {
|
|
if ctx.Value(needParsedSqlInCtx) != nil {
|
|
return ctx
|
|
}
|
|
return context.WithValue(ctx, needParsedSqlInCtx, true)
|
|
}
|
|
|
|
func (d *Driver) getNeedParsedSqlFromCtx(ctx context.Context) bool {
|
|
if ctx.Value(needParsedSqlInCtx) != nil {
|
|
return true
|
|
}
|
|
return false
|
|
}
|