English | 简体中文
[TDengine] 提供了 GO 驱动程序 [taosSql
][driver-go],实现了 GO 语言的内置数据库操作接口 database/sql/driver
。
v2 与 v3 版本不兼容,与 TDengine 版本对应如下:
driver-go 版本 | TDengine 版本 | 主要功能 |
---|---|---|
v3.6.0 | 3.3.5.0及以上 | stmt2 原生写入、DSN 支持密码包含特殊字符(url.QueryEscape) |
v3.5.8 | 3.3.2.0及以上 / 3.1.2.0及以上 | 修复空指针异常 |
v3.5.7 | 3.3.2.0及以上 / 3.1.2.0及以上 | taosWS 和 taosRestful 支持传入 request id |
v3.5.6 | 3.3.2.0及以上 / 3.1.2.0及以上 | 提升 websocket 查询和写入性能 |
v3.5.5 | 3.2.3.0及以上 / 3.1.1.27及以上 | restful 支持跳过 ssl 证书检查 |
v3.5.4 | 3.2.3.0及以上 / 3.1.1.27及以上 | 兼容 TDengine 3.3.0.0 tmq raw data |
v3.5.3 | 3.2.3.0及以上 / 3.1.1.27及以上 | 重构 taosWS |
v3.5.2 | 3.2.3.0及以上 / 3.1.1.27及以上 | websocket 压缩和优化消息订阅性能 |
v3.5.1 | 3.2.1.0及以上 / 3.1.1.13及以上 | 原生 stmt 查询和 geometry 类型支持 |
v3.5.0 | 3.0.5.0及以上 | 获取消费进度及按照指定进度开始消费 |
v3.3.1 | 3.0.4.1及以上 | 基于 websocket 的 schemaless 协议写入 |
v3.1.0 | 3.0.2.2及以上 | 提供贴近 kafka 的订阅 api |
v3.0.4 | 3.0.2.2及以上 | 新增 request id 相关接口 |
v3.0.3 | 3.0.1.5及以上 | 基于 websocket 的 statement 写入 |
v3.0.2 | 3.0.1.5及以上 | 基于 websocket 的数据查询和写入 |
v3.0.1 | 3.0.0.0及以上 | 基于 websocket 的消息订阅 |
v3.0.0 | 3.0.0.0及以上 | 适配 TDengine 3.0 查询和写入 |
对新建项目,建议使用 Go 1.14+,并使用 GO Modules 方式进行管理。
go mod init taos-demo
引入 taosSql:
import (
"database/sql"
_ "github.com/taosdata/driver-go/v3/taosSql"
)
使用 go mod
方式管理依赖包:
go mod tidy
或通过 go get
直接下载安装:
go get github.com/taosdata/driver-go/v3/taosSql
TDengine Go 连接器提供 database/sql 标准接口,使用方法简单示例如下:
package main
import (
"database/sql"
"fmt"
"time"
_ "github.com/taosdata/driver-go/v3/taosSql"
)
func main() {
var taosUri = "root:taosdata@tcp(localhost:6030)/"
taos, err := sql.Open("taosSql", taosUri)
if err != nil {
fmt.Println("failed to connect TDengine, err:", err)
return
}
defer taos.Close()
taos.Exec("create database if not exists test")
taos.Exec("use test")
taos.Exec("create table if not exists tb1 (ts timestamp, a int)")
_, err = taos.Exec("insert into tb1 values(now, 0)(now+1s,1)(now+2s,2)(now+3s,3)")
if err != nil {
fmt.Println("failed to insert, err:", err)
return
}
rows, err := taos.Query("select * from tb1")
if err != nil {
fmt.Println("failed to select from table, err:", err)
return
}
defer rows.Close()
for rows.Next() {
var r struct {
ts time.Time
a int
}
err := rows.Scan(&r.ts, &r.a)
if err != nil {
fmt.Println("scan error:\n", err)
return
}
fmt.Println(r.ts, r.a)
}
}
常用 API 列表:
-
sql.Open(DRIVER_NAME string, dataSourceName string) *DB
该 API 用来创建
database/sql
DB 对象,类型为*DB
,DRIVER_NAME 设置为字符串taosSql
, dataSourceName 设置为字符串user:password@tcp(host:port)/dbname
,对应于 TDengine 的高可用机制,可以使用user:password@cfg(/etc/taos)/dbname
来使用/etc/taos/taos.cfg
中的多 EP 配置。注意: 该 API 成功创建的时候,并没有做权限等检查,只有在真正执行 Query 或者 Exec 的时候才能真正的去创建连接,并同时检查 user/password/host/port 是不是合法。 另外,由于整个驱动程序大部分实现都下沉到 taosSql 所依赖的 libtaos 中。所以,sql.Open 本身特别轻量。
-
func (db *DB) Exec(query string, args ...interface{}) (Result, error)
sql.Open 内置的方法,用来执行非查询相关 SQL,如
create
,alter
等。 -
func (db *DB) Query(query string, args ...interface{}) (*Rows, error)
sql.Open 内置的方法,用来执行查询语句。
-
func (db *DB) Close() error
sql.Open 内置的方法,关闭 DB 对象。
创建消费:
func NewConsumer(conf *tmq.ConfigMap) (*Consumer, error)
订阅单个主题:
func (c *Consumer) Subscribe(topic string, rebalanceCb RebalanceCb) error
订阅:
func (c *Consumer) SubscribeTopics(topics []string, rebalanceCb RebalanceCb) error
轮询消息:
func (c *Consumer) Poll(timeoutMs int) tmq.Event
提交消息:
func (c *Consumer) Commit() ([]tmq.TopicPartition, error)
获取消费进度:
func (c *Consumer) Assignment() (partitions []tmq.TopicPartition, err error)
按照指定的进度消费:
func (c *Consumer) Seek(partition tmq.TopicPartition, ignoredTimeoutMs int) error
取消订阅:
func (c *Consumer) Unsubscribe() error
关闭消费:
func (c *Consumer) Close() error
示例代码:examples/tmq/main.go
。
InfluxDB 格式:
func (conn *Connector) InfluxDBInsertLines(lines []string, precision string) error
示例代码:examples/schemaless/influx/main.go
。
OpenTSDB telnet 格式:
func (conn *Connector) OpenTSDBInsertTelnetLines(lines []string) error
示例代码:examples/schemaless/telnet/main.go
。
OpenTSDB json 格式:
func (conn *Connector) OpenTSDBInsertJsonPayload(payload string) error
示例代码:examples/schemaless/json/main.go
。
prepare sql:
func (stmt *InsertStmt) Prepare(sql string) error
设置子表名:
func (stmt *InsertStmt) SetSubTableName(name string) error
设置表名:
func (stmt *InsertStmt) SetTableName(name string) error
设置子表名和标签:
func (stmt *InsertStmt) SetTableNameWithTags(tableName string, tags *param.Param) error
绑定参数:
func (stmt *InsertStmt) BindParam(params []*param.Param, bindType *param.ColumnType) error
添加批次:
func (stmt *InsertStmt) AddBatch() error
执行:
func (stmt *InsertStmt) Execute() error
获取影响行数:
func (stmt *InsertStmt) GetAffectedRows() int
关闭 stmt:
func (stmt *InsertStmt) Close() error
示例代码:examples/stmtinsert/main.go
。
通过 restful 方式实现 database/sql
接口,使用方法简单示例如下:
package main
import (
"database/sql"
"fmt"
"time"
_ "github.com/taosdata/driver-go/v3/taosRestful"
)
func main() {
var taosDSN = "root:taosdata@http(localhost:6041)/"
taos, err := sql.Open("taosRestful", taosDSN)
if err != nil {
fmt.Println("failed to connect TDengine, err:", err)
return
}
defer taos.Close()
taos.Exec("create database if not exists test")
taos.Exec("create table if not exists test.tb1 (ts timestamp, a int)")
_, err = taos.Exec("insert into test.tb1 values(now, 0)(now+1s,1)(now+2s,2)(now+3s,3)")
if err != nil {
fmt.Println("failed to insert, err:", err)
return
}
rows, err := taos.Query("select * from test.tb1")
if err != nil {
fmt.Println("failed to select from table, err:", err)
return
}
defer rows.Close()
for rows.Next() {
var r struct {
ts time.Time
a int
}
err := rows.Scan(&r.ts, &r.a)
if err != nil {
fmt.Println("scan error:\n", err)
return
}
fmt.Println(r.ts, r.a)
}
}
引入
import (
"database/sql"
_ "github.com/taosdata/driver-go/v3/taosRestful"
)
sql.Open
的 driverName 为 taosRestful
DSN 格式为:
数据库用户名:数据库密码@连接方式(域名或 ip:端口)/[数据库][?参数]
样例:
root:taosdata@http(localhost:6041)/test?readBufferSize=52428800
参数:
disableCompression
是否接受压缩数据,默认为true
不接受压缩数据,如果传输数据使用 gzip 压缩设置为false
。readBufferSize
读取数据的缓存区大小默认为 4K (4096),当查询结果数据量多时可以适当调大该值。skipVerify
是否跳过 ssl 证书检查,默认为false
不跳过,如果使用 https 连接可以设置为true
。
由于 restful 接口无状态所以 use db
语法不会生效,需要将 db 名称放到 sql 语句中,如:create table if not exists tb1 (ts timestamp, a int)
改为 create table if not exists test.tb1 (ts timestamp, a int)
否则将报错 [0x217] Database not specified or available
也可以将 db 名称放到 DSN 中,将 root:taosdata@http(localhost:6041)/
改为 root:taosdata@http(localhost:6041)/test
。当指定的 db 不存在时执行 create database
语句不会报报错,而执行针对该 db 的其他查询或写入操作会报错。完整示例如下:
package main
import (
"database/sql"
"fmt"
"time"
_ "github.com/taosdata/driver-go/v3/taosRestful"
)
func main() {
var taosDSN = "root:taosdata@http(localhost:6041)/test"
taos, err := sql.Open("taosRestful", taosDSN)
if err != nil {
fmt.Println("failed to connect TDengine, err:", err)
return
}
defer taos.Close()
taos.Exec("create database if not exists test")
taos.Exec("create table if not exists tb1 (ts timestamp, a int)")
_, err = taos.Exec("insert into tb1 values(now, 0)(now+1s,1)(now+2s,2)(now+3s,3)")
if err != nil {
fmt.Println("failed to insert, err:", err)
return
}
rows, err := taos.Query("select * from tb1")
if err != nil {
fmt.Println("failed to select from table, err:", err)
return
}
defer rows.Close()
for rows.Next() {
var r struct {
ts time.Time
a int
}
err := rows.Scan(&r.ts, &r.a)
if err != nil {
fmt.Println("scan error:\n", err)
return
}
fmt.Println(r.ts, r.a)
}
}
通过 websocket 方式实现 database/sql
接口,使用方法简单示例如下:
package main
import (
"database/sql"
"fmt"
"time"
_ "github.com/taosdata/driver-go/v3/taosWS"
)
func main() {
var taosDSN = "root:taosdata@ws(localhost:6041)/"
taos, err := sql.Open("taosWS", taosDSN)
if err != nil {
fmt.Println("failed to connect TDengine, err:", err)
return
}
defer taos.Close()
taos.Exec("create database if not exists test")
taos.Exec("create table if not exists test.tb1 (ts timestamp, a int)")
_, err = taos.Exec("insert into test.tb1 values(now, 0)(now+1s,1)(now+2s,2)(now+3s,3)")
if err != nil {
fmt.Println("failed to insert, err:", err)
return
}
rows, err := taos.Query("select * from test.tb1")
if err != nil {
fmt.Println("failed to select from table, err:", err)
return
}
defer rows.Close()
for rows.Next() {
var r struct {
ts time.Time
a int
}
err := rows.Scan(&r.ts, &r.a)
if err != nil {
fmt.Println("scan error:\n", err)
return
}
fmt.Println(r.ts, r.a)
}
}
引入
import (
"database/sql"
_ "github.com/taosdata/driver-go/v3/taosWS"
)
sql.Open
的 driverName 为 taosWS
DSN 格式为:
数据库用户名:数据库密码@连接方式(域名或 ip:端口)/[数据库][?参数]
样例:
root:taosdata@ws(localhost:6041)/test?writeTimeout=10s&readTimeout=10m
参数:
writeTimeout
通过 websocket 发送数据的超时时间。readTimeout
通过 websocket 接收响应数据的超时时间。enableCompression
是否压缩传输数据,默认为false
不发送压缩数据。
通过 websocket 方式使用 tmq。服务端需要启动 taoAdapter。
func NewConfig(url string, chanLength uint) *Config
创建配置项,传入 websocket 地址和发送管道长度。
func (c *Config) SetConnectUser(user string) error
设置用户名。
func (c *Config) SetConnectPass(pass string) error
设置密码。
func (c *Config) SetClientID(clientID string) error
设置客户端标识。
func (c *Config) SetGroupID(groupID string) error
设置订阅组 ID。
func (c *Config) SetWriteWait(writeWait time.Duration) error
设置发送消息等待时间。
func (c *Config) SetMessageTimeout(timeout time.Duration) error
设置消息超时时间。
func (c *Config) SetErrorHandler(f func(consumer *Consumer, err error))
设置错误处理方法。
func (c *Config) SetCloseHandler(f func())
设置关闭处理方法。
func NewConsumer(conf *tmq.ConfigMap) (*Consumer, error)
创建消费者。
func (c *Consumer) Subscribe(topic string, rebalanceCb RebalanceCb) error
订阅单个主题。
func (c *Consumer) SubscribeTopics(topics []string, rebalanceCb RebalanceCb) error
订阅主题。
func (c *Consumer) Poll(timeoutMs int) tmq.Event
轮询消息。
func (c *Consumer) Commit() ([]tmq.TopicPartition, error)
提交消息。
func (c *Consumer) Assignment() (partitions []tmq.TopicPartition, err error)
获取消费进度。
func (c *Consumer) Seek(partition tmq.TopicPartition, ignoredTimeoutMs int) error
按照指定的进度消费。
func (c *Consumer) Close() error
关闭连接。
示例代码:examples/tmqoverws/main.go
。
通过 websocket 方式使用 stmt。服务端需要启动 taoAdapter。
-
func NewConfig(url string, chanLength uint) *Config
创建配置项,传入 websocket 地址和发送管道长度。
-
func (c *Config) SetCloseHandler(f func())
设置关闭处理方法。
-
func (c *Config) SetConnectDB(db string) error
设置连接 DB。
-
func (c *Config) SetConnectPass(pass string) error
设置连接密码。
-
func (c *Config) SetConnectUser(user string) error
设置连接用户名。
-
func (c *Config) SetErrorHandler(f func(connector *Connector, err error))
设置错误处理函数。
-
func (c *Config) SetMessageTimeout(timeout time.Duration) error
设置消息超时时间。
-
func (c *Config) SetWriteWait(writeWait time.Duration) error
设置发送消息等待时间。
-
func NewConnector(config *Config) (*Connector, error)
创建连接。
-
func (c *Connector) Init() (*Stmt, error)
初始化参数。
-
func (c *Connector) Close() error
关闭连接。
-
func (s *Stmt) Prepare(sql string) error
参数绑定预处理 SQL 语句。
-
func (s *Stmt) SetTableName(name string) error
参数绑定设置表名。
-
func (s *Stmt) SetTags(tags *param.Param, bindType *param.ColumnType) error
参数绑定设置标签。
-
func (s *Stmt) BindParam(params []*param.Param, bindType *param.ColumnType) error
参数绑定多行数据。
-
func (s *Stmt) AddBatch() error
添加到参数绑定批处理。
-
func (s *Stmt) Exec() error
执行参数绑定。
-
func (s *Stmt) GetAffectedRows() int
获取参数绑定插入受影响行数。
-
func (s *Stmt) Close() error
结束参数绑定。
完整参数绑定示例参见 GitHub 示例文件
driver-go
├── af //高级功能
├── common //通用方法以及常量
├── errors //错误类型
├── examples //样例
├── taosRestful // 数据库操作标准接口 (restful)
├── taosSql // 数据库操作标准接口
├── types // 内置类型
├── wrapper // cgo 包装器
└── ws // websocket
driver-go: https://github.com/taosdata/driver-go
TDengine: https://github.com/taosdata/TDengine