用Go构建一个SQL解析器

开发 开发工具
在本文中,小编将向大家简单介绍如何在 Go 中构造 LL(1) 解析器,并应用于解析SQL查询。希望大家能用 Go 对简单的解析器算法有一个了解和简单应用

摘要

本文旨在简单介绍如何在 Go 中构造 LL(1) 解析器,在本例中用于解析SQL查询。

为了简单起见,我们将处理子选择、函数、复杂嵌套表达式和所有 SQL 风格都支持的其他特性。这些特性与我们将要使用的策略紧密相关。

1分钟理论

一个解析器包含两个部分:

  • 词法分析:也就是“Tokeniser”
  • 语法分析:AST 的创建

词法分析

让我们用例子来定义一下。“Tokenising”以下查询:

  1. SELECT id, name FROM 'users.csv' 

表示提取构成此查询的“tokens”。tokeniser 的结果像这样:

  1. []string{"SELECT", "id", ",", "name", "FROM", "'users.csv'"} 

语法分析

这部分实际上是我们查看 tokens 的地方,确保它们有意义并解析它们来构造出一些结构体,以一种对将要使用它的应用程序更方便的方式表示查询(例如,用于执行查询,用颜色高亮显示它)。在这一步之后,我们会得到这样的结果:

  1. query{ 
  2.     Type: "Select", 
  3.     TableName: "users.csv", 
  4.     Fields: ["id", "name"], 

有很多原因可能会导致解析失败,所以同时执行这两个步骤可能会比较方便,并在出现错误时可以立即停止。

策略

我们将定义一个像这样的解析器:

  1. type parser struct { 
  2.   sql             string        // The query to parse 
  3.   i               int           // Where we are in the query 
  4.   query           query.Query   // The "query struct" we'll build 
  5.   step            step          // What's this? Read on... 
  6.  
  7. // Main function that returns the "query struct" or an error 
  8. func (p *parser) Parse() (query.Query, error) {} 
  9.  
  10. // A "look-ahead" function that returns the next token to parse 
  11. func (p *parser) peek() (string) {} 
  12.  
  13. // same as peek(), but advancing our "i" index 
  14. func (p *parser) pop() (string) {} 

直观地说,我们首先要做的是“peek() ***个 token”。在基础的SQL语法中,只有几个有效的初始 token:SELECT、UPDATE、DELETE等;其他的都是错误的。代码像这样:

  1. switch strings.ToUpper(parser.peek()) { 
  2.  
  3. case "SELECT": 
  4.   parser.query.type = "SELECT" // start building the "query struct" 
  5.   parser.pop() 
  6.   // TODO continue with SELECT query parsing... 
  7.  
  8. case "UPDATE": 
  9.   // TODO handle UPDATE 
  10.  
  11. // TODO other cases... 
  12.  
  13. default: 
  14.   return parser.query, fmt.Errorf("invalid query type") 
  15.  

我们基本上可以填写 TODO 和让它跑起来!然而,聪明的读者会发现,解析整个 SELECT 查询的代码很快会变得混乱,而且我们有许多类型的查询需要解析。所以我们需要一些结构。

有限状态机

FSMs 是一个非常有趣的话题,但我们来这里不是为了讲这个,所以不会深入介绍。让我们只关注我们需要什么。

在我们的解析过程中,在任何给定的点(与其说“点”,不如称其称为“节点”),只有少数 token 是有效的,在找到这些 token 之后,我们将进入新的节点,其中不同的 token 是有效的,以此类推,直到完成对查询的解析。我们可以将这些节点关系可视化为有向图:

点转换可以用一个更简单的表来定义,但是:

我们可以直接将这个表转换成一个非常大的 switch 语句。我们将使用那个我们之前定义过的 parser.step 属性:

  1. func (p *parser) Parse() (query.Query, error) { 
  2.   parser.step = stepType // initial step 
  3.  
  4.   for parser.i < len(parser.sql) { 
  5.     nextToken :parser.peek() 
  6.  
  7.     switch parser.step { 
  8.     case stepType: 
  9.       switch nextToken { 
  10.       case UPDATE: 
  11.         parser.query.type = "UPDATE" 
  12.         parser.step = stepUpdateTable 
  13.  
  14.       // TODO cases of other query types 
  15.       } 
  16.     case stepUpdateSet: 
  17.       // ... 
  18.     case stepUpdateField: 
  19.       // ... 
  20.     case stepUpdateComma: 
  21.       // ... 
  22.     } 
  23.  
  24.     parser.pop() 
  25.   } 
  26.  
  27.   return parser.query, nil 

好了!注意,有些步骤可能会有条件地循环回以前的步骤,比如 SELECT 字段定义上的逗号。这种策略对于基本的解析器非常适用。然而,随着语法变得复杂,状态的数量将急剧增加,因此编写起来可能会变得单调乏味。我建议在编写代码时进行测试;更多信息请见下文。

Peek() 实现

记住,我们需要同时实现 peek() 和 pop() 。因为它们几乎是一样的,所以我们用一个辅助函数来保持代码整洁。此外,pop() 应该进一步推进索引,以避免取到空格。

  1. func (p *parser) peek() string { 
  2.   peeked, _ :p.peekWithLength() 
  3.   return peeked 
  4.  
  5. func (p *parser) pop() string { 
  6.   peeked, len :p.peekWithLength() 
  7.   p.i += len 
  8.   p.popWhitespace() 
  9.   return peeked 
  10.  
  11. func (p *parser) popWhitespace() { 
  12.   for ; p.i < len(p.sql) && p.sql[p.i] == ' '; p.i++ { 
  13.   } 

下面是我们可能想要得到的令牌列表:

  1. var reservedWords = []string{ 
  2.   "(", ")", ">=", "<=", "!=", ",", "=", ">", "<", 
  3.   "SELECT", "INSERT INTO", "VALUES", "UPDATE", 
  4.   "DELETE FROM", "WHERE", "FROM", "SET", 

除此之外,我们可能会遇到带引号的字符串或纯标识符(例如字段名)。下面是一个完整的 peekWithLength() 实现:

  1. func (p *parser) peekWithLength() (string, int) { 
  2.   if p.i >= len(p.sql) { 
  3.     return "", 0 
  4.   } 
  5.   for _, rWord :range reservedWords { 
  6.     token :p.sql[p.i:min(len(p.sql), p.i+len(rWord))] 
  7.     upToken :strings.ToUpper(token) 
  8.     if upToken == rWord { 
  9.       return upToken, len(upToken) 
  10.     } 
  11.   } 
  12.   if p.sql[p.i] == '\'' { // Quoted string 
  13.     return p.peekQuotedStringWithLength() 
  14.   } 
  15.   return p.peekIdentifierWithLength() 

其余的函数都很简单,留给读者作为练习。如果您感兴趣,可以查看 github 的链接,其中包含完整的源代码实现。

最终验证

解析器可能会在得到完整的查询定义之前找到字符串的末尾。实现一个 parser.validate() 函数可能是一个好主意,该函数查看生成的“query”结构,如果它不完整或错误,则返回一个错误。

测试Go的表格驱动测试模式非常适合我们的情况:

  1. type testCase struct { 
  2.   Name     string         // description of the test 
  3.   SQL      string         // input sql e.g. "SELECT a FROM 'b'" 
  4.   Expected query.Query    // expected resulting "query" struct 
  5.   Err      error          // expected error result 

测试实例:

  1. ts := []testCase{ 
  2.     { 
  3.       Name:     "empty query fails", 
  4.       SQL:      "", 
  5.       Expected: query.Query{}, 
  6.       Err:      fmt.Errorf("query type cannot be empty"), 
  7.     }, 
  8.     { 
  9.       Name:     "SELECT without FROM fails", 
  10.       SQL:      "SELECT", 
  11.       Expected: query.Query{Type: query.Select}, 
  12.       Err:      fmt.Errorf("table name cannot be empty"), 
  13.     }, 
  14.     ... 

像这样测试测试用例:

  1. for _, tc :range ts { 
  2.     t.Run(tc.Name, func(t *testing.T) { 
  3.       actual, err :Parse(tc.SQL) 
  4.       if tc.Err != nil && err == nil { 
  5.         t.Errorf("Error should have been %v", tc.Err) 
  6.       } 
  7.       if tc.Err == nil && err != nil { 
  8.         t.Errorf("Error should have been nil but was %v", err) 
  9.       } 
  10.       if tc.Err != nil && err != nil { 
  11.         require.Equal(t, tc.Err, err, "Unexpected error") 
  12.       } 
  13.       if len(actual) > 0 { 
  14.         require.Equal(t, tc.Expected, actual[0], 
  15.           "Query didn't match expectation") 
  16.       } 
  17.     }) 
  18.   } 

我使用 verify 是因为当查询结构不匹配时,它提供了一个 diff 输出。

深入理解

这个实验非常适合:

  • 学习 LL(1) 解析器算法
  • 自定义解析无依赖关系的简单语法

然而,这种方法可能会变得单调乏味,而且有一定的局限性。考虑一下如何解析任意复杂的复合表达式(例如 sqrt(a) =(1 *(2 + 3)))。

要获得更强大的解析模型,请查看解析器组合符。goyacc 是一个流行的Go实现。

下面是完整的解析器地址(或点击阅读原文查看):http://github.com/marianogappa/sqlparser

 

责任编辑:赵宁宁 来源: 51CTO专栏
相关推荐

2022-10-20 11:00:52

SQL解析器

2017-02-14 10:20:43

Java Class解析器

2022-06-28 08:17:10

JSON性能反射

2023-12-30 13:33:36

Python解析器JSON

2014-05-15 09:45:58

Python解析器

2011-11-28 15:40:52

wiresharkRDP解析器

2015-02-10 14:32:37

XSS漏洞XSS

2023-07-25 14:24:33

元素JSX解析器

2024-01-08 08:36:29

HTTPGo代理服务器

2023-05-10 08:05:41

GoWeb应用

2014-10-15 11:01:02

Web应用测试应用

2022-09-20 08:43:37

Go编程语言Web

2009-06-19 11:42:09

Scala计算器解析

2021-08-27 12:16:34

fastjarJAR文件Java

2018-03-19 17:40:10

Python区块链

2011-04-01 16:16:27

JavaScript

2009-03-19 09:26:05

RSS解析器MagpieRSS

2021-04-15 08:55:51

Go struc代码

2021-04-25 08:58:00

Go拍照云盘

2018-04-12 12:45:02

数据库MySQL构建架构
点赞
收藏

51CTO技术栈公众号