使用go連接clickhouse方式
前言
近段時(shí)間業(yè)務(wù)在一個(gè)局點(diǎn)測試clickhouse,用java寫的代碼在環(huán)境上一直連接不上clickhouse服務(wù),報(bào)錯(cuò)信息也比較奇怪,No client available
,研發(fā)查了一段時(shí)間沒查出來,讓運(yùn)維這邊繼續(xù)查:
運(yùn)維同學(xué)查了各種監(jiān)聽配置,防火墻這些,都沒什么問題,但是沒有明確證據(jù)能夠提供證明通過http方式能訪問到數(shù)據(jù)庫,時(shí)間拖得比較久,項(xiàng)目上就急了,讓盡快找到問題,所以就用go寫了個(gè)小工具拉到集群上試試看8123這個(gè)端口到底能不能正常提供服務(wù)。
正文
先安裝必要的庫,clickhouse官方提供了2個(gè)版本的庫,v1和v2,v1版本已經(jīng)明確不會(huì)繼續(xù)更新了,所以用新不用舊哈,可以用官方庫的方式或者用dsn的方式,這個(gè)我下面一起說,安裝庫的命令:
go get github.com/ClickHouse/clickhouse-go/v2
構(gòu)造結(jié)構(gòu)體
編寫結(jié)構(gòu)體,存放基本信息:
type Clickhouse struct { Host string // 服務(wù)端主機(jī) Port int // 端口 DB string // 數(shù)據(jù)庫 User string // 用戶名 Password string // 密碼 Connection *sql.DB // 建立連接后存放連接 Rows *sql.Rows // 運(yùn)行sql后的結(jié)果存放 }
Connection
主要是用來建立連接后把相關(guān)信息存放,這樣方便繼續(xù)調(diào)用其他的方法,因?yàn)槲业闹饕康氖菧y試數(shù)據(jù)庫能否連通和運(yùn)行Sql,所以這里Rows
用來存放測試的select語句的結(jié)果。
參數(shù)讀取
這塊沒什么好說的,連接的參數(shù)直接從命令行讀取,用flag包就好:
var ( host = flag.String("host", "localhost", "clickhouse host") port = flag.Int("port", 8123, "clickhouse port") user = flag.String("user", "default", "clichouse user") pass = flag.String("password", "", "clickhouse password") db = flag.String("db", "default", "clickhouse database") query = flag.String("query", "show tables", "query you will run") mode = flag.String("mode", "driver", "driver or dsn") )
前面幾個(gè)參數(shù)不用解釋,主要是query
和mode
,query
是要運(yùn)行的sql語句,我們默認(rèn)就認(rèn)為跑的是select語句,然后是mode
,允許選擇模式,用戶可以使用driver
或者dsn
兩種模式進(jìn)行連接,我寫了兩個(gè)不同的方法,其實(shí)也可以在一個(gè)Connect方法里做判斷,看個(gè)人習(xí)慣;
建立連接
接下來我們建立數(shù)據(jù)庫連接:
// func (c *Clickhouse) Conn() { c.Connection = clickhouse.OpenDB(&clickhouse.Options{ Addr: []string{fmt.Sprintf("%s:%d", c.Host, c.Port)}, Auth: clickhouse.Auth{ Database: c.DB, Username: c.User, Password: c.Password, }, Settings: clickhouse.Settings{ "max_execution_time": 60, }, DialTimeout: 5 * time.Second, Compression: &clickhouse.Compression{ Method: clickhouse.CompressionBrotli, Level: 5, }, // 必須添加協(xié)議方式 Protocol: clickhouse.HTTP, }) } func (c *Clickhouse) ConnDsn() { conn, err := sql.Open("clickhouse", fmt.Sprintf("http://%s:%d/%s?username=%s&password=%s", c.Host, c.Port, c.DB, c.User, c.Password)) if err != nil { log.Printf("Connect to the server failed, %s.\n", err.Error()) return } c.Connection = conn }
參考官網(wǎng)的實(shí)例,實(shí)現(xiàn)兩種連接方式,關(guān)閉方法就直接把sql.DB和sql.Rows都關(guān)閉就可以了:
func (c *Clickhouse) Close() { c.Connection.Close() c.Rows.Close() }
發(fā)起查詢
查詢使用Query方法進(jìn)行:
func (c *Clickhouse) Select(query string) { rows, err := c.Connection.Query(query) if err != nil { log.Printf("Query select failed, %s.\n", err.Error()) return } c.Rows = rows }
查詢的結(jié)果我保存到Rows里,方便后面的解析
結(jié)果解析
比較麻煩的就是結(jié)果的解析了,用過database/sql
庫的哥們都知道,這個(gè)庫只提供了基礎(chǔ)的一些接口,查詢出來一般用Scan去獲取數(shù)據(jù),用法類似這樣:
問題就在于,Scan要指定和sql查詢出來一樣多的變量,對(duì)于我們這個(gè)小工具來說,sql是不一定的,所以查詢出來的字段數(shù)量肯定yes不定的,如何動(dòng)態(tài)處理這個(gè)問題,肯定是不能直接寫一個(gè)結(jié)構(gòu)體解決的,先看我的代碼:
func (c *Clickhouse) Show() { cols, err := c.Rows.Columns() if err != nil { log.Printf("Failed to get table columns, %s.\n", err.Error()) return } // 一行數(shù)據(jù),使用any是為了避開數(shù)據(jù)類型的問題 var rows = make([]any, len(cols)) // 存實(shí)際的值,是byte數(shù)組,長度以列的數(shù)量為準(zhǔn) var values = make([][]byte, len(cols)) for i := 0; i < len(cols); i++ { rows[i] = &values[i] } // 打印表頭 fmt.Println(strings.Join(cols, ",")) for c.Rows.Next() { if err = c.Rows.Scan(rows...); err != nil { fmt.Println(err) return } var vString []string for _, v := range values { vString = append(vString, string(v)) } // 逐行打印出來 fmt.Println(strings.Join(vString, ",")) } }
大概思路是這樣:
- Scan需要傳入每個(gè)用來綁定單行數(shù)據(jù)值的變量,所以values是實(shí)際存儲(chǔ)數(shù)據(jù)的byte數(shù)組,然后把數(shù)組的每個(gè)元素的地址再存入到rows數(shù)組中;
- 現(xiàn)在可以用rows[index]這樣的方式來訪問values中的值了,把rows直接作為入?yún)魅氲絊can,在每次循環(huán)中,把values的值轉(zhuǎn)成逗號(hào)分割的字符串,直接打印
結(jié)果驗(yàn)證
OK,現(xiàn)在邏輯完成了,我們運(yùn)行測試一下,
go run main.go -host hostname -password paswword -query "select * from clusters" -db system -mode dsn
只查詢2個(gè)字段,2行數(shù)據(jù):
結(jié)語
完成,然后把工具放到生產(chǎn)環(huán)境一測試,查詢都正常,這下開發(fā)哥們要繼續(xù)查他的程序問題了
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
用gin開發(fā)的golang項(xiàng)目三種開發(fā)模式方式
這篇文章主要介紹了用gin開發(fā)的golang項(xiàng)目三種開發(fā)模式方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01go語言調(diào)用c語言的so動(dòng)態(tài)庫的實(shí)現(xiàn)
在Go語言開發(fā)過程中,有時(shí)需要調(diào)用C或C++編寫的so動(dòng)態(tài)庫,本文介紹了如何在Go語言中調(diào)用so庫的步驟和注意事項(xiàng),包括環(huán)境準(zhǔn)備、編譯生成.so文件、Go文件編寫、以及可能遇到的問題和解決方法,感興趣的可以了解一下2024-10-10Go語言中如何確保Cookie數(shù)據(jù)的安全傳輸
這篇文章主要介紹了Go語言中如何確保Cookie數(shù)據(jù)的安全傳輸,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03如何使用Go語言獲取當(dāng)天、昨天、明天、某天0點(diǎn)時(shí)間戳以及格式化時(shí)間
這篇文章主要給大家介紹了關(guān)于如何使用Go語言獲取當(dāng)天、昨天、明天、某天0點(diǎn)時(shí)間戳以及格式化時(shí)間的相關(guān)資料,格式化時(shí)間戳是將時(shí)間戳轉(zhuǎn)換為特定的日期和時(shí)間格式,文中通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下2023-10-10