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>
571 lines
20 KiB
Go
571 lines
20 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
|
||
|
||
import (
|
||
"context"
|
||
"fmt"
|
||
"testing"
|
||
"time"
|
||
|
||
"github.com/google/uuid"
|
||
"github.com/shopspring/decimal"
|
||
|
||
"github.com/gogf/gf/v2/database/gdb"
|
||
"github.com/gogf/gf/v2/frame/g"
|
||
"github.com/gogf/gf/v2/os/gtime"
|
||
"github.com/gogf/gf/v2/test/gtest"
|
||
"github.com/gogf/gf/v2/util/gconv"
|
||
"github.com/gogf/gf/v2/util/grand"
|
||
"github.com/gogf/gf/v2/util/guid"
|
||
)
|
||
|
||
const (
|
||
sqlVisitsDDL = `
|
||
CREATE TABLE IF NOT EXISTS visits (
|
||
id UInt64,
|
||
duration Float64,
|
||
url String,
|
||
created DateTime
|
||
) ENGINE = MergeTree()
|
||
PRIMARY KEY id
|
||
ORDER BY id
|
||
`
|
||
dimSqlDDL = `
|
||
CREATE TABLE IF NOT EXISTS dim (
|
||
"code" String COMMENT '编码',
|
||
"translation" String COMMENT '译文',
|
||
"superior" UInt64 COMMENT '上级ID',
|
||
"row_number" UInt16 COMMENT '行号',
|
||
"is_active" UInt8 COMMENT '是否激活',
|
||
"is_preset" UInt8 COMMENT '是否预置',
|
||
"category" String COMMENT '类别',
|
||
"tree_path" Array(String) COMMENT '树路径',
|
||
"id" UInt64 COMMENT '代理主键ID',
|
||
"scd" UInt64 COMMENT '缓慢变化维ID',
|
||
"version" UInt64 COMMENT 'Merge版本ID',
|
||
"sign" Int8 COMMENT '标识位',
|
||
"created_by" UInt64 COMMENT '创建者ID',
|
||
"created_at" DateTime64(3,'Asia/Shanghai') COMMENT '创建时间',
|
||
"updated_by" UInt64 COMMENT '最后修改者ID',
|
||
"updated_at" DateTime64(3,'Asia/Shanghai') COMMENT '最后修改时间',
|
||
"updated_tick" UInt16 COMMENT '累计修改次数'
|
||
) ENGINE = ReplacingMergeTree("version")
|
||
ORDER BY ("id","scd")
|
||
COMMENT '会计准则';
|
||
`
|
||
dimSqlDML = `
|
||
insert into dim (code, translation, superior, row_number, is_active, is_preset, category, tree_path, id, scd, version, sign, created_by, created_at, updated_by, updated_at, updated_tick)
|
||
values ('CN', '{"zh_CN":"中国大陆会计准则","en_US":"Chinese mainland accounting legislation"}', 0, 1, 1, 1, 1, '[''CN'']', 607972403489804288, 0, 0, 0, 607536279118155777, '2017-09-06 00:00:00', 607536279118155777, '2017-09-06 00:00:00', 0),
|
||
('HK', '{"zh_CN":"中国香港会计准则","en_US":"Chinese Hong Kong accounting legislation"}', 0, 2, 1, 1, 1, '[''HK'']', 607972558544834566, 0, 0, 0, 607536279118155777, '2017-09-06 00:00:00', 607536279118155777, '2017-09-06 00:00:00', 0);
|
||
`
|
||
factSqlDDL = `
|
||
CREATE TABLE IF NOT EXISTS fact (
|
||
"adjustment_level" UInt64 COMMENT '调整层ID',
|
||
"data_version" UInt64 COMMENT '数据版本ID',
|
||
"accounting_legislation" UInt64 COMMENT '会计准则ID',
|
||
"fiscal_year" UInt16 COMMENT '会计年度',
|
||
"fiscal_period" UInt8 COMMENT '会计期间',
|
||
"fiscal_year_period" UInt32 COMMENT '会计年度期间',
|
||
"legal_entity" UInt64 COMMENT '法人主体ID',
|
||
"cost_center" UInt64 COMMENT '成本中心ID',
|
||
"legal_entity_partner" UInt64 COMMENT '内部关联方ID',
|
||
"financial_posting" UInt64 COMMENT '凭证头ID',
|
||
"line" UInt16 COMMENT '行号',
|
||
"general_ledger_account" UInt64 COMMENT '总账科目ID',
|
||
"debit" Decimal64(9) COMMENT '借方金额',
|
||
"credit" Decimal64(9) COMMENT '贷方金额',
|
||
"transaction_currency" UInt64 COMMENT '交易币种ID',
|
||
"debit_tc" Decimal64(9) COMMENT '借方金额(交易币种)',
|
||
"credit_tc" Decimal64(9) COMMENT '贷方金额(交易币种)',
|
||
"posting_date" Date32 COMMENT '过账日期',
|
||
"gc_year" UInt16 COMMENT '公历年',
|
||
"gc_quarter" UInt8 COMMENT '公历季',
|
||
"gc_month" UInt8 COMMENT '公历月',
|
||
"gc_week" UInt8 COMMENT '公历周',
|
||
"raw_info" String COMMENT '源信息',
|
||
"summary" String COMMENT '摘要',
|
||
"id" UInt64 COMMENT '代理主键ID',
|
||
"version" UInt64 COMMENT 'Merge版本ID',
|
||
"sign" Int8 COMMENT '标识位'
|
||
) ENGINE = ReplacingMergeTree("version")
|
||
ORDER BY ("adjustment_level","data_version","legal_entity","fiscal_year","fiscal_period","financial_posting","line")
|
||
PARTITION BY ("adjustment_level","data_version","legal_entity","fiscal_year","fiscal_period")
|
||
COMMENT '数据主表';
|
||
`
|
||
factSqlDML = `
|
||
insert into fact (adjustment_level, data_version, accounting_legislation, fiscal_year, fiscal_period, fiscal_year_period, legal_entity, cost_center, legal_entity_partner, financial_posting, line, general_ledger_account, debit, credit, transaction_currency, debit_tc, credit_tc, posting_date, gc_year, gc_quarter, gc_month, gc_week, raw_info, summary, id, version, sign)
|
||
values (607970943242866688, 607973669943119880, 607972403489804288, 2022, 3, 202203, 607974511316307985, 0, 607976190010986520, 607996702456025136, 1, 607985607569838111, 8674.39, 0, 607974898261823505, 8674.39, 0, '2022-03-05', 2022, 1, 3, 11, '{}', '摘要', 607992882741121073, 0, 0),
|
||
(607970943242866688, 607973669943119880, 607972403489804288, 2022, 4, 202204, 607974511316307985, 0, 607976190010986520, 607993586419503145, 1, 607985607569838111, 9999.88, 0, 607974898261823505, 9999.88, 0, '2022-04-10', 2022, 2, 4, 18, '{}', '摘要', 607996939140599857, 0, 0);
|
||
`
|
||
expmSqlDDL = `
|
||
CREATE TABLE IF NOT EXISTS data_type (
|
||
Col1 UInt8 COMMENT '列1'
|
||
, Col2 Nullable(String) COMMENT '列2'
|
||
, Col3 FixedString(3) COMMENT '列3'
|
||
, Col4 String COMMENT '列4'
|
||
, Col5 Map(String, UInt8) COMMENT '列5'
|
||
, Col6 Array(String) COMMENT '列6'
|
||
, Col7 Tuple(String, UInt8, Array(Map(String, String))) COMMENT '列7'
|
||
, Col8 DateTime COMMENT '列8'
|
||
, Col9 UUID COMMENT '列9'
|
||
, Col10 DateTime COMMENT '列10'
|
||
, Col11 Decimal(9, 2) COMMENT '列11'
|
||
, Col12 Decimal(9, 2) COMMENT '列12'
|
||
) ENGINE = MergeTree()
|
||
PRIMARY KEY Col4
|
||
ORDER BY Col4
|
||
`
|
||
)
|
||
|
||
func clickhouseConfigDB() gdb.DB {
|
||
connect, err := gdb.New(gdb.ConfigNode{
|
||
Host: "127.0.0.1",
|
||
Port: "9000",
|
||
User: "default",
|
||
Name: "default",
|
||
Type: "clickhouse",
|
||
Debug: false,
|
||
})
|
||
gtest.AssertNil(err)
|
||
gtest.AssertNE(connect, nil)
|
||
return connect
|
||
}
|
||
|
||
func clickhouseLink() gdb.DB {
|
||
connect, err := gdb.New(gdb.ConfigNode{
|
||
Link: "clickhouse:default:@tcp(127.0.0.1:9000)/default?dial_timeout=200ms&max_execution_time=60",
|
||
})
|
||
gtest.AssertNil(err)
|
||
gtest.AssertNE(connect, nil)
|
||
return connect
|
||
}
|
||
|
||
func createClickhouseTableVisits(connect gdb.DB) error {
|
||
_, err := connect.Exec(context.Background(), sqlVisitsDDL)
|
||
return err
|
||
}
|
||
|
||
func createClickhouseTableDim(connect gdb.DB) error {
|
||
_, err := connect.Exec(context.Background(), dimSqlDDL)
|
||
return err
|
||
}
|
||
|
||
func createClickhouseTableFact(connect gdb.DB) error {
|
||
_, err := connect.Exec(context.Background(), factSqlDDL)
|
||
return err
|
||
}
|
||
|
||
func createClickhouseExampleTable(connect gdb.DB) error {
|
||
_, err := connect.Exec(context.Background(), expmSqlDDL)
|
||
return err
|
||
}
|
||
|
||
func dropClickhouseTableVisits(conn gdb.DB) {
|
||
sqlStr := fmt.Sprintf("DROP TABLE IF EXISTS `visits`")
|
||
_, _ = conn.Exec(context.Background(), sqlStr)
|
||
}
|
||
|
||
func dropClickhouseTableDim(conn gdb.DB) {
|
||
sqlStr := fmt.Sprintf("DROP TABLE IF EXISTS `dim`")
|
||
_, _ = conn.Exec(context.Background(), sqlStr)
|
||
}
|
||
|
||
func dropClickhouseTableFact(conn gdb.DB) {
|
||
sqlStr := fmt.Sprintf("DROP TABLE IF EXISTS `fact`")
|
||
_, _ = conn.Exec(context.Background(), sqlStr)
|
||
}
|
||
|
||
func dropClickhouseExampleTable(conn gdb.DB) {
|
||
sqlStr := fmt.Sprintf("DROP TABLE IF EXISTS `data_type`")
|
||
_, _ = conn.Exec(context.Background(), sqlStr)
|
||
}
|
||
|
||
func TestDriverClickhouse_Create(t *testing.T) {
|
||
gtest.AssertNil(createClickhouseTableVisits(clickhouseConfigDB()))
|
||
}
|
||
|
||
func TestDriverClickhouse_New(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNE(connect, nil)
|
||
gtest.AssertNil(connect.PingMaster())
|
||
gtest.AssertNil(connect.PingSlave())
|
||
}
|
||
|
||
func TestDriverClickhouse_OpenLink_Ping(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNE(connect, nil)
|
||
gtest.AssertNil(connect.PingMaster())
|
||
}
|
||
|
||
func TestDriverClickhouse_Tables(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableVisits(connect), nil)
|
||
defer dropClickhouseTableVisits(connect)
|
||
tables, err := connect.Tables(context.Background())
|
||
gtest.AssertNil(err)
|
||
gtest.AssertNE(len(tables), 0)
|
||
}
|
||
|
||
func TestDriverClickhouse_TableFields_Use_Config(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseTableVisits(connect))
|
||
defer dropClickhouseTableVisits(connect)
|
||
field, err := connect.TableFields(context.Background(), "visits")
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(len(field), 4)
|
||
gtest.AssertNQ(field, nil)
|
||
}
|
||
|
||
func TestDriverClickhouse_TableFields_Use_Link(t *testing.T) {
|
||
connect := clickhouseLink()
|
||
gtest.AssertNil(createClickhouseTableVisits(connect))
|
||
defer dropClickhouseTableVisits(connect)
|
||
field, err := connect.TableFields(context.Background(), "visits")
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(len(field), 4)
|
||
gtest.AssertNQ(field, nil)
|
||
}
|
||
|
||
func TestDriverClickhouse_Transaction(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
defer dropClickhouseTableVisits(connect)
|
||
gtest.AssertNE(connect.Transaction(context.Background(), func(ctx context.Context, tx gdb.TX) error {
|
||
return nil
|
||
}), nil)
|
||
}
|
||
|
||
func TestDriverClickhouse_InsertIgnore(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
_, err := connect.InsertIgnore(context.Background(), "", nil)
|
||
gtest.AssertEQ(err, errUnsupportedInsertIgnore)
|
||
}
|
||
|
||
func TestDriverClickhouse_InsertAndGetId(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
_, err := connect.InsertAndGetId(context.Background(), "", nil)
|
||
gtest.AssertEQ(err, errUnsupportedInsertGetId)
|
||
}
|
||
|
||
func TestDriverClickhouse_InsertOne(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableVisits(connect), nil)
|
||
defer dropClickhouseTableVisits(connect)
|
||
_, err := connect.Model("visits").Data(g.Map{
|
||
"duration": float64(grand.Intn(999)),
|
||
"url": gconv.String(grand.Intn(999)),
|
||
"created": time.Now(),
|
||
}).Insert()
|
||
gtest.AssertNil(err)
|
||
}
|
||
|
||
func TestDriverClickhouse_InsertMany(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableVisits(connect), nil)
|
||
defer dropClickhouseTableVisits(connect)
|
||
tx, err := connect.Begin(context.Background())
|
||
gtest.AssertEQ(err, errUnsupportedBegin)
|
||
gtest.AssertNil(tx)
|
||
}
|
||
|
||
func TestDriverClickhouse_Insert(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableVisits(connect), nil)
|
||
defer dropClickhouseTableVisits(connect)
|
||
type insertItem struct {
|
||
Id uint64 `orm:"id"`
|
||
Duration float64 `orm:"duration"`
|
||
Url string `orm:"url"`
|
||
Created time.Time `orm:"created"`
|
||
}
|
||
var (
|
||
insertUrl = "https://goframe.org"
|
||
total = 0
|
||
item = insertItem{
|
||
Duration: 1,
|
||
Url: insertUrl,
|
||
Created: time.Now(),
|
||
}
|
||
)
|
||
_, err := connect.Model("visits").Data(item).Insert()
|
||
gtest.AssertNil(err)
|
||
_, err = connect.Model("visits").Data(item).Save()
|
||
gtest.AssertNil(err)
|
||
total, err = connect.Model("visits").Count()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(total, 2)
|
||
var list []*insertItem
|
||
for i := 0; i < 50; i++ {
|
||
list = append(list, &insertItem{
|
||
Duration: float64(grand.Intn(999)),
|
||
Url: insertUrl,
|
||
Created: time.Now(),
|
||
})
|
||
}
|
||
_, err = connect.Model("visits").Data(list).Insert()
|
||
gtest.AssertNil(err)
|
||
_, err = connect.Model("visits").Data(list).Save()
|
||
gtest.AssertNil(err)
|
||
total, err = connect.Model("visits").Count()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(total, 102)
|
||
}
|
||
|
||
func TestDriverClickhouse_Insert_Use_Exec(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableFact(connect), nil)
|
||
defer dropClickhouseTableFact(connect)
|
||
_, err := connect.Exec(context.Background(), factSqlDML)
|
||
gtest.AssertNil(err)
|
||
}
|
||
|
||
func TestDriverClickhouse_Delete(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableVisits(connect), nil)
|
||
defer dropClickhouseTableVisits(connect)
|
||
_, err := connect.Model("visits").Where("created >", "2021-01-01 00:00:00").Delete()
|
||
gtest.AssertNil(err)
|
||
_, err = connect.Model("visits").
|
||
Where("created >", "2021-01-01 00:00:00").
|
||
Where("duration > ", 0).
|
||
Where("url is not null").
|
||
Delete()
|
||
gtest.AssertNil(err)
|
||
}
|
||
|
||
func TestDriverClickhouse_Update(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableVisits(connect), nil)
|
||
defer dropClickhouseTableVisits(connect)
|
||
_, err := connect.Model("visits").Where("created > ", "2021-01-01 15:15:15").Data(g.Map{
|
||
"created": time.Now().Format("2006-01-02 15:04:05"),
|
||
}).Update()
|
||
gtest.AssertNil(err)
|
||
_, err = connect.Model("visits").
|
||
Where("created > ", "2021-01-01 15:15:15").
|
||
Where("duration > ", 0).
|
||
Where("url is not null").
|
||
Data(g.Map{
|
||
"created": time.Now().Format("2006-01-02 15:04:05"),
|
||
}).Update()
|
||
}
|
||
|
||
func TestDriverClickhouse_Replace(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
_, err := connect.Replace(context.Background(), "", nil)
|
||
gtest.AssertEQ(err, errUnsupportedReplace)
|
||
}
|
||
|
||
func TestDriverClickhouse_DoFilter(t *testing.T) {
|
||
rawSQL := "select * from visits where 1 = 1"
|
||
this := Driver{}
|
||
replaceSQL, _, err := this.DoFilter(context.Background(), nil, rawSQL, []interface{}{1})
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(rawSQL, replaceSQL)
|
||
|
||
// this SQL can't run ,clickhouse will report an error because there is no WHERE statement
|
||
rawSQL = "update visit set url = '1'"
|
||
replaceSQL, _, err = this.DoFilter(context.Background(), nil, rawSQL, []interface{}{1})
|
||
gtest.AssertNil(err)
|
||
|
||
// this SQL can't run ,clickhouse will report an error because there is no WHERE statement
|
||
rawSQL = "delete from visit"
|
||
replaceSQL, _, err = this.DoFilter(context.Background(), nil, rawSQL, []interface{}{1})
|
||
gtest.AssertNil(err)
|
||
|
||
ctx := this.injectNeedParsedSql(context.Background())
|
||
rawSQL = "UPDATE visit SET url = '1' WHERE url = '0'"
|
||
replaceSQL, _, err = this.DoFilter(ctx, nil, rawSQL, []interface{}{1})
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(replaceSQL, "ALTER TABLE visit UPDATE url = '1' WHERE url = '0'")
|
||
|
||
rawSQL = "DELETE FROM visit WHERE url = '0'"
|
||
replaceSQL, _, err = this.DoFilter(ctx, nil, rawSQL, []interface{}{1})
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(replaceSQL, "ALTER TABLE visit DELETE WHERE url = '0'")
|
||
}
|
||
|
||
func TestDriverClickhouse_Select(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseTableVisits(connect))
|
||
defer dropClickhouseTableVisits(connect)
|
||
_, err := connect.Model("visits").Data(g.Map{
|
||
"url": "goframe.org",
|
||
"duration": float64(1),
|
||
}).Insert()
|
||
gtest.AssertNil(err)
|
||
temp, err := connect.Model("visits").Where("url", "goframe.org").Where("duration >= ", 1).One()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(temp.IsEmpty(), false)
|
||
_, err = connect.Model("visits").Data(g.Map{
|
||
"url": "goframe.org",
|
||
"duration": float64(2),
|
||
}).Insert()
|
||
gtest.AssertNil(err)
|
||
data, err := connect.Model("visits").Where("url", "goframe.org").Where("duration >= ", 1).All()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(len(data), 2)
|
||
}
|
||
|
||
func TestDriverClickhouse_Exec_OPTIMIZE(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseTableVisits(connect))
|
||
defer dropClickhouseTableVisits(connect)
|
||
sqlStr := "OPTIMIZE table visits"
|
||
_, err := connect.Exec(context.Background(), sqlStr)
|
||
gtest.AssertNil(err)
|
||
}
|
||
|
||
func TestDriverClickhouse_ExecInsert(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertEQ(createClickhouseTableDim(connect), nil)
|
||
defer dropClickhouseTableDim(connect)
|
||
_, err := connect.Exec(context.Background(), dimSqlDML)
|
||
gtest.AssertNil(err)
|
||
}
|
||
|
||
func TestDriverClickhouse_NilTime(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseExampleTable(connect))
|
||
defer dropClickhouseExampleTable(connect)
|
||
type testNilTime struct {
|
||
Col1 uint8
|
||
Col2 string
|
||
Col3 string
|
||
Col4 string
|
||
Col5 map[string]uint8
|
||
Col6 []string
|
||
Col7 []interface{}
|
||
Col8 *time.Time
|
||
Col9 uuid.UUID
|
||
Col10 *gtime.Time
|
||
Col11 decimal.Decimal
|
||
Col12 *decimal.Decimal
|
||
}
|
||
insertData := []*testNilTime{}
|
||
money := decimal.NewFromFloat(1.12)
|
||
strMoney, _ := decimal.NewFromString("99999.999")
|
||
for i := 0; i < 10000; i++ {
|
||
insertData = append(insertData, &testNilTime{
|
||
Col4: "Inc.",
|
||
Col9: uuid.New(),
|
||
Col7: []interface{}{ // Tuple(String, UInt8, Array(Map(String, String)))
|
||
"String Value", uint8(5), []map[string]string{
|
||
map[string]string{"key": "value"},
|
||
map[string]string{"key": "value"},
|
||
map[string]string{"key": "value"},
|
||
}},
|
||
Col11: money,
|
||
Col12: &strMoney,
|
||
})
|
||
}
|
||
_, err := connect.Model("data_type").Data(insertData).Insert()
|
||
gtest.AssertNil(err)
|
||
count, err := connect.Model("data_type").Where("Col4", "Inc.").Count()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(count, 10000)
|
||
|
||
data, err := connect.Model("data_type").Where("Col4", "Inc.").One()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertNE(data, nil)
|
||
g.Dump(data)
|
||
gtest.AssertEQ(data["Col11"].String(), "1.12")
|
||
gtest.AssertEQ(data["Col12"].String(), "99999.99")
|
||
}
|
||
|
||
func TestDriverClickhouse_BatchInsert(t *testing.T) {
|
||
// example from
|
||
// https://github.com/ClickHouse/clickhouse-go/blob/v2/examples/std/batch/main.go
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseExampleTable(connect))
|
||
defer dropClickhouseExampleTable(connect)
|
||
insertData := []g.Map{}
|
||
for i := 0; i < 10000; i++ {
|
||
insertData = append(insertData, g.Map{
|
||
"Col1": uint8(42),
|
||
"Col2": "ClickHouse",
|
||
"Col3": "Inc",
|
||
"Col4": guid.S(),
|
||
"Col5": map[string]uint8{"key": 1}, // Map(String, UInt8)
|
||
"Col6": []string{"Q", "W", "E", "R", "T", "Y"}, // Array(String)
|
||
"Col7": []interface{}{ // Tuple(String, UInt8, Array(Map(String, String)))
|
||
"String Value", uint8(5), []map[string]string{
|
||
map[string]string{"key": "value"},
|
||
map[string]string{"key": "value"},
|
||
map[string]string{"key": "value"},
|
||
},
|
||
},
|
||
"Col8": gtime.Now(),
|
||
"Col9": uuid.New(),
|
||
"Col10": nil,
|
||
})
|
||
}
|
||
_, err := connect.Model("data_type").Data(insertData).Insert()
|
||
gtest.AssertNil(err)
|
||
count, err := connect.Model("data_type").Where("Col2", "ClickHouse").Where("Col3", "Inc").Count()
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(count, 10000)
|
||
}
|
||
|
||
func TestDriverClickhouse_Open(t *testing.T) {
|
||
// link
|
||
// DSM
|
||
// clickhouse://username:password@host1:9000,host2:9000/database?dial_timeout=200ms&max_execution_time=60
|
||
link := "clickhouse://default@127.0.0.1:9000,127.0.0.1:9000/default?dial_timeout=200ms&max_execution_time=60"
|
||
db, err := gdb.New(gdb.ConfigNode{
|
||
Link: link,
|
||
Type: "clickhouse",
|
||
})
|
||
gtest.AssertNil(err)
|
||
gtest.AssertNil(db.PingMaster())
|
||
}
|
||
|
||
func TestDriverClickhouse_TableFields(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseExampleTable(connect))
|
||
defer dropClickhouseExampleTable(connect)
|
||
dataTypeTable, err := connect.TableFields(context.Background(), "data_type")
|
||
gtest.AssertNil(err)
|
||
gtest.AssertNE(dataTypeTable, nil)
|
||
|
||
var result = map[string][]interface{}{
|
||
"Col1": {0, "Col1", "UInt8", false, "", "", "", "列1"},
|
||
"Col2": {1, "Col2", "String", true, "", "", "", "列2"},
|
||
"Col3": {2, "Col3", "FixedString(3)", false, "", "", "", "列3"},
|
||
"Col4": {3, "Col4", "String", false, "", "", "", "列4"},
|
||
"Col5": {4, "Col5", "Map(String, UInt8)", false, "", "", "", "列5"},
|
||
"Col6": {5, "Col6", "Array(String)", false, "", "", "", "列6"},
|
||
"Col7": {6, "Col7", "Tuple(String, UInt8, Array(Map(String, String)))", false, "", "", "", "列7"},
|
||
"Col8": {7, "Col8", "DateTime", false, "", "", "", "列8"},
|
||
"Col9": {8, "Col9", "UUID", false, "", "", "", "列9"},
|
||
"Col10": {9, "Col10", "DateTime", false, "", "", "", "列10"},
|
||
"Col11": {10, "Col11", "Decimal(9, 2)", false, "", "", "", "列11"},
|
||
"Col12": {11, "Col12", "Decimal(9, 2)", false, "", "", "", "列12"},
|
||
}
|
||
for k, v := range result {
|
||
_, ok := dataTypeTable[k]
|
||
gtest.AssertEQ(ok, true)
|
||
gtest.AssertEQ(dataTypeTable[k].Index, v[0])
|
||
gtest.AssertEQ(dataTypeTable[k].Name, v[1])
|
||
gtest.AssertEQ(dataTypeTable[k].Type, v[2])
|
||
gtest.AssertEQ(dataTypeTable[k].Null, v[3])
|
||
gtest.AssertEQ(dataTypeTable[k].Key, v[4])
|
||
gtest.AssertEQ(dataTypeTable[k].Default, v[5])
|
||
gtest.AssertEQ(dataTypeTable[k].Comment, v[7])
|
||
}
|
||
}
|
||
|
||
func TestDriverClickhouse_TableFields_HasField(t *testing.T) {
|
||
connect := clickhouseConfigDB()
|
||
gtest.AssertNil(createClickhouseExampleTable(connect))
|
||
defer dropClickhouseExampleTable(connect)
|
||
// 未修复前:panic: runtime error: index out of range [12] with length 12
|
||
b, err := connect.GetCore().HasField(context.Background(), "data_type", "Col1")
|
||
gtest.AssertNil(err)
|
||
gtest.AssertEQ(b, true)
|
||
}
|