golang使用TiDB的SQL Parse对SQL指纹化

复制代码
//mkdir sqlaprse && cd sqlaprse
//export GOPROXY=https://proxy.golang.com.cn,direct
//go mod init sqlaprse && touch main.go
//go get -v github.com/pingcap/tidb/parser@40b72e7
//go get github.com/fatih/color
//go get github.com/pingcap/tidb/types/parser_driver@40b72e7
//go run main.go -sql="select * from t1 where id=1 and date>'2023-08-18'"
package main

import (
	"bytes"
	"flag"
	"fmt"
	"github.com/fatih/color"
	"github.com/pingcap/tidb/parser"
	"github.com/pingcap/tidb/parser/ast"
	"github.com/pingcap/tidb/parser/format"
	driver "github.com/pingcap/tidb/types/parser_driver"
	"sync" // 导入 sync 包用于 WaitGroup
)

type FingerprintVisitor struct{}

func (f *FingerprintVisitor) Enter(n ast.Node) (node ast.Node, skipChildren bool) {
	if v, ok := n.(*driver.ValueExpr); ok {
		v.SetValue([]byte("?"))
	}
	return n, false
}

func (f *FingerprintVisitor) Leave(n ast.Node) (node ast.Node, ok bool) {
	return n, true
}

var (
	Success *color.Color
	Err     *color.Color
	sql     string
)

func processSQL(sql string, wg *sync.WaitGroup) {
	defer wg.Done() // 通知 WaitGroup 此 goroutine 已完成
	p := parser.New()
	stmt, err := p.ParseOneStmt(sql, "", "")
	if err != nil {
		Err.Println("解析错误:" + err.Error())
		return
	}
	stmt.Accept(&FingerprintVisitor{})

	buf := new(bytes.Buffer)
	restoreCtx := format.NewRestoreCtx(format.RestoreKeyWordUppercase|format.RestoreNameBackQuotes, buf)
	err = stmt.Restore(restoreCtx)
	if nil != err {
		Err.Println("解析错误:" + err.Error())
		return
	}
	Success.Println(buf.String())
}

func main() {
	Success = color.New(color.FgHiGreen, color.Bold, color.Underline)
	Err = color.New(color.FgHiRed, color.Bold)
	flag.StringVar(&sql, "sql", "", "SQL语句")
	flag.Parse()

	// 分割 SQL 语句
	statements := splitSQLStatements(sql)

	// 创建 WaitGroup 以等待所有 goroutine 完成
	var wg sync.WaitGroup

	for _, stmt := range statements {
		wg.Add(1) // 为每个 goroutine 增加 WaitGroup 计数
		go processSQL(stmt, &wg)
	}

	// 等待所有 goroutine 完成
	wg.Wait()
}

// 用于分割 SQL 语句的辅助函数
func splitSQLStatements(sql string) []string {
	// 在此实现分割 SQL 语句的逻辑
	// 例如,如果每个语句位于单独的行上并以分号结尾,则可以按分号分割
	// 并根据需要修改此函数。这是为了演示目的的简化示例。
	// 您可能需要处理边缘情况并改进分割逻辑。
	statements := []string{sql}
	return statements
}
相关推荐
数据知道6 分钟前
PostgreSQL 性能优化:连接数过多的原因分析与连接池方案
数据库·postgresql·性能优化
怣506 分钟前
MySQL子查询实战指南:数据操作(增删改查)与通用表达式
数据库·chrome·mysql
范纹杉想快点毕业9 分钟前
从单片机基础到程序框架:构建嵌入式系统的完整路径
数据库·mongodb
数据知道12 分钟前
PostgreSQL性能优化:如何定期清理无用索引以释放磁盘空间(索引膨胀监控)
数据库·postgresql·性能优化
喵叔哟13 分钟前
67.【.NET8 实战--孢子记账--从单体到微服务--转向微服务】--新增功能--分摊功能总体设计与业务流程
数据库·微服务·架构
tryCbest13 分钟前
Oracle查看存储过程
数据库·oracle
咩咩不吃草20 分钟前
【MySQL】表和列、增删改查语句及数据类型约束详解
数据库·mysql·语法
不懒不懒20 分钟前
【MySQL 实战:从零搭建规范用户表(含完整 SQL 与避坑指南)】
数据库
ID_1800790547324 分钟前
Python结合淘宝关键词API进行商品价格监控与预警
服务器·数据库·python
数据知道38 分钟前
PostgreSQL 故障排查:万字详解如何找出数据库中的死锁
数据库·postgresql