SQL 语句

SQL 语法

FROM、SELECT 和 WHERE 子句:

规则引擎的 SQL 语句基本格式为:

  1. SELECT <字段名> FROM <主题> [WHERE <条件>]
  • FROM 子句将规则挂载到某个主题上
  • SELECT 子句用于对数据进行变换,并选择出感兴趣的字段
  • WHERE 子句用于对 SELECT 选择出来的某个字段施加条件过滤

FOREACH、DO 和 INCASE 子句:

如果对于一个数组数据,想针对数组中的每个元素分别执行一些操作并执行 Actions,需要使用 FOREACH-DO-INCASE 语法。其基本格式为:

  1. FOREACH <字段名> [DO <条件>] [INCASE <条件>] FROM <主题> [WHERE <条件>]
  • FOREACH 子句用于选择需要做 foreach 操作的字段,注意选择出的字段必须为数组类型
  • DO 子句用于对 FOREACH 选择出来的数组中的每个元素进行变换,并选择出感兴趣的字段
  • INCASE 子句用于对 DO 选择出来的某个字段施加条件过滤

其中 DO 和 INCASE 子句都是可选的。DO 相当于针对当前循环中对象的 SELECT 子句,而 INCASE 相当于针对当前循环中对象的 WHERE 语句。

事件和事件主题

规则引擎的 SQL 语句既可以处理消息(消息发布),也可以处理事件(客户端上下线、客户端订阅等)。对于消息,FROM 子句后面直接跟主题名;对于事件,FROM 子句后面跟事件主题。

事件消息的主题以 "$events/" 开头,比如 "$events/client_connected", "$events/session_subscribed"。 如果想让 emqx 将事件消息发布出来,可以在 emqx_rule_engine.conf 文件中配置。

所有支持的事件及其可用字段详见: 规则事件

SQL 语句示例:

基本语法举例

  • 从 topic 为 “t/a” 的消息中提取所有字段:
  1. SELECT * FROM "t/a"
  • 从 topic 为 “t/a” 或 “t/b” 的消息中提取所有字段:
  1. SELECT * FROM "t/a","t/b"
  • 从 topic 能够匹配到 ‘t/#’ 的消息中提取所有字段。
  1. SELECT * FROM "t/#"
  • 从 topic 能够匹配到 ‘t/#’ 的消息中提取 qos, username 和 clientid 字段:
  1. SELECT qos, username, clientid FROM "t/#"
  • 从任意 topic 的消息中提取 username 字段,并且筛选条件为 username = ‘Steven’:
  1. SELECT username FROM "#" WHERE username='Steven'
  • 从任意 topic 的 JSON 消息体(payload) 中提取 x 字段,并创建别名 x 以便在 WHERE 子句中使用。WHERE 子句限定条件为 x = 1。下面这个 SQL 语句可以匹配到消息体 {“x”: 1}, 但不能匹配到消息体 {“x”: 2}:
  1. SELECT payload as p FROM "#" WHERE p.x = 1
  • 类似于上面的 SQL 语句,但嵌套地提取消息体中的数据,下面的 SQL 语句可以匹配到 JSON 消息体 {“x”: {“y”: 1}}:
  1. SELECT payload as a FROM "#" WHERE a.x.y = 1
  • 在 clientid = ‘c1’ 尝试连接时,提取其来源 IP 地址和端口号:
  1. SELECT peername as ip_port FROM "$events/client_connected" WHERE clientid = 'c1'
  • 筛选所有订阅 ‘t/#’ 主题且订阅级别为 QoS1 的 clientid:
  1. SELECT clientid FROM "$events/session_subscribed" WHERE topic = 't/#' and qos = 1
  • 筛选所有订阅主题能匹配到 ‘t/#’ 且订阅级别为 QoS1 的 clientid。注意与上例不同的是,这里用的是主题匹配操作符 ‘=~’,所以会匹配订阅 ‘t’ 或 ‘t/+/a’ 的订阅事件:
  1. SELECT clientid FROM "$events/session_subscribed" WHERE topic =~ 't/#' and qos = 1

TIP

  • FROM 子句后面的主题需要用双引号 "" 引起来。
  • WHERE 子句后面接筛选条件,如果使用到字符串需要用单引号 '' 引起来。
  • FROM 子句里如有多个主题,需要用逗号 "," 分隔。例如 SELECT * FROM “t/1”, “t/2” 。
  • 可以使用使用 "." 符号对 payload 进行嵌套选择。

遍历语法(FOREACH-DO-INCASE) 举例

假设有 ClientID 为 c_steve、主题为 t/1 的消息,消息体为 JSON 格式,其中 sensors 字段为包含多个 Object 的数组:

  1. {
  2. "date": "2020-04-24",
  3. "sensors": [
  4. {"name": "a", "idx":0},
  5. {"name": "b", "idx":1},
  6. {"name": "c", "idx":2}
  7. ]
  8. }

示例1: 要求将 sensors 里的各个对象,分别作为数据输入重新发布消息到 sensors/${idx} 主题,内容为 ${name}。即最终规则引擎将会发出 3 条消息:

  1. 主题:sensors/0 内容:a
  2. 主题:sensors/1 内容:b
  3. 主题:sensors/2 内容:c

要完成这个规则,我们需要配置如下动作:

  • 动作类型:消息重新发布 (republish)
  • 目的主题:sensors/${idx}
  • 目的 QoS:0
  • 消息内容模板:${name}

以及如下 SQL 语句:

  1. FOREACH
  2. payload.sensors
  3. FROM "t/#"

示例解析:

这个 SQL 中,FOREACH 子句指定需要进行遍历的数组 sensors,则选取结果为:

  1. [
  2. {
  3. "name": "a",
  4. "idx": 0
  5. },
  6. {
  7. "name": "b",
  8. "idx": 1
  9. },
  10. {
  11. "name": "c",
  12. "idx": 2
  13. }
  14. ]

FOREACH 语句将会对于结果数组里的每个对象分别执行 “消息重新发布” 动作,所以将会执行重新发布动作 3 次。

示例2: 要求将 sensors 里的 idx 值大于或等于 1 的对象,分别作为数据输入重新发布消息到 sensors/${idx} 主题,内容为 clientid=${clientid},name=${name},date=${date}。即最终规则引擎将会发出 2 条消息:

  1. 主题:sensors/1 内容:clientid=c_steve,name=b,date=2020-04-24
  2. 主题:sensors/2 内容:clientid=c_steve,name=c,date=2020-04-24

要完成这个规则,我们需要配置如下动作:

  • 动作类型:消息重新发布 (republish)
  • 目的主题:sensors/${idx}
  • 目的 QoS:0
  • 消息内容模板:clientid=${clientid},name=${name},date=${date}

以及如下 SQL 语句:

  1. FOREACH
  2. payload.sensors
  3. DO
  4. clientid,
  5. item.name as name,
  6. item.idx as idx
  7. INCASE
  8. item.idx >= 1
  9. FROM "t/#"

示例解析:

这个 SQL 中,FOREACH 子句指定需要进行遍历的数组 sensors; DO 子句选取每次操作需要的字段,这里我们选了外层的 clientid 字段,以及当前 sensor 对象的 nameidx 两个字段,注意 item 代表 sensors 数组中本次循环的对象。INCASE 子句是针对 DO 语句中字段的筛选条件,仅仅当 idx >= 1 满足条件。所以 SQL 的选取结果为:

  1. [
  2. {
  3. "name": "b",
  4. "idx": 1,
  5. "clientid": "c_emqx"
  6. },
  7. {
  8. "name": "c",
  9. "idx": 2,
  10. "clientid": "c_emqx"
  11. }
  12. ]

FOREACH 语句将会对于结果数组里的每个对象分别执行 “消息重新发布” 动作,所以将会执行重新发布动作 2 次。

在 DO 和 INCASE 语句里,可以使用 item 访问当前循环的对象,也可以通过在 FOREACH 使用 as 语法自定义一个变量名。所以本例中的 SQL 语句又可以写为:

  1. FOREACH
  2. payload.sensors as s
  3. DO
  4. clientid,
  5. s.name as name,
  6. s.idx as idx
  7. INCASE
  8. s.idx >= 1
  9. FROM "t/#"

示例3: 在示例2 的基础上,去掉 clientid 字段 c_steve 中的 c_ 前缀

在 FOREACH 和 DO 语句中可以调用各类 SQL 函数,若要将 c_steve 变为 steve,则可以把例2 中的 SQL 改为:

  1. FOREACH
  2. payload.sensors as s
  3. DO
  4. nth(2, tokens(clientid,'_')) as clientid,
  5. s.name as name,
  6. s.idx as idx
  7. INCASE
  8. s.idx >= 1
  9. FROM "t/#"

另外,FOREACH 子句中也可以放多个表达式,只要最后一个表达式是指定要遍历的数组即可。比如我们将消息体改一下,sensors 外面多套一层 Object:

  1. {
  2. "date": "2020-04-24",
  3. "data": {
  4. "sensors": [
  5. {"name": "a", "idx":0},
  6. {"name": "b", "idx":1},
  7. {"name": "c", "idx":2}
  8. ]
  9. }
  10. }

则 FOREACH 中可以在决定要遍历的数组之前把 data 选取出来:

  1. FOREACH
  2. payload.data as data
  3. data.sensors as s
  4. ...

CASE-WHEN 语法示例

示例1: 将消息中 x 字段的值范围限定在 0~7 之间。

  1. SELECT
  2. CASE WHEN payload.x < 0 THEN 0
  3. WHEN payload.x > 7 THEN 7
  4. ELSE payload.x
  5. END as x
  6. FROM "t/#"

假设消息为:

  1. {"x": 8}

则上面的 SQL 输出为:

  1. {"x": 7}

FROM 子句可用的事件主题

事件主题名释义
$events/message_delivered消息投递
$events/message_acked消息确认
$events/message_dropped消息丢弃
$events/client_connected连接完成
$events/client_disconnected连接断开
$events/session_subscribed订阅
$events/session_unsubscribed取消订阅

SELECT 和 WHERE 子句可用的字段

SELECT 和 WHERE 子句可用的字段与事件的类型相关。其中 clientid, usernameevent 是通用字段,每种事件类型都有。

普通主题 (消息发布)

event事件类型,固定为 “message.publish”
idMQTT 消息 ID
clientidClient ID
username用户名
payloadMQTT 消息体
peerhost客户端的 IPAddress
topicMQTT 主题
qosMQTT 消息的 QoS
flagsMQTT 消息的 Flags
headersMQTT 消息内部与流程处理相关的额外数据
timestamp事件触发时间 (ms)
publish_received_atPUBLISH 消息到达 Broker 的时间 (ms)
node事件触发所在节点

$events/message_delivered (消息投递)

event事件类型,固定为 “message.delivered”
idMQTT 消息 ID
from_clientid消息来源 Client ID
from_username消息来源用户名
clientid消息目的 Client ID
username消息目的用户名
payloadMQTT 消息体
peerhost客户端的 IPAddress
topicMQTT 主题
qosMQTT 消息的 QoS
flagsMQTT 消息的 Flags
timestamp事件触发时间 (ms)
publish_received_atPUBLISH 消息到达 Broker 的时间 (ms)
node事件触发所在节点

$events/message_acked (消息确认)

event事件类型,固定为 “message.acked”
idMQTT 消息 ID
from_clientid消息来源 Client ID
from_username消息来源用户名
clientid消息目的 Client ID
username消息目的用户名
payloadMQTT 消息体
peerhost客户端的 IPAddress
topicMQTT 主题
qosMQTT 消息的 QoS
flagsMQTT 消息的 Flags
timestamp事件触发时间 (ms)
publish_received_atPUBLISH 消息到达 Broker 的时间 (ms)
node事件触发所在节点

$events/message_dropped (消息丢弃)

event事件类型,固定为 “message.dropped”
idMQTT 消息 ID
reason消息丢弃原因
clientid消息目的 Client ID
username消息目的用户名
payloadMQTT 消息体
peerhost客户端的 IPAddress
topicMQTT 主题
qosMQTT 消息的 QoS
flagsMQTT 消息的 Flags
timestamp事件触发时间 (ms)
publish_received_atPUBLISH 消息到达 Broker 的时间 (ms)
node事件触发所在节点

$events/client_connected (终端连接成功)

event事件类型,固定为 “client.connected”
clientid消息目的 Client ID
username消息目的用户名
mountpoint主题挂载点(主题前缀)
peername终端的 IPAddress 和 Port
socknameemqx 监听的 IPAddress 和 Port
proto_name协议名字
proto_ver协议版本
keepaliveMQTT 保活间隔
clean_startMQTT clean_start
expiry_intervalMQTT Session 过期时间
is_bridge是否为 MQTT bridge 连接
connected_at终端连接完成时间 (s)
timestamp事件触发时间 (ms)
node事件触发所在节点

$events/client_disconnected (终端连接断开)

event事件类型,固定为 “client.disconnected”
reason终端连接断开原因
clientid消息目的 Client ID
username消息目的用户名
peername终端的 IPAddress 和 Port
socknameemqx 监听的 IPAddress 和 Port
disconnected_at终端连接断开时间 (s)
timestamp事件触发时间 (ms)
node事件触发所在节点

$events/session_subscribed (终端订阅成功)

event事件类型,固定为 “session.subscribed”
clientid消息目的 Client ID
username消息目的用户名
peerhost客户端的 IPAddress
topicMQTT 主题
qosMQTT 消息的 QoS
timestamp事件触发时间 (ms)
node事件触发所在节点

$events/session_unsubscribed (取消终端订阅成功)

event事件类型,固定为 “session.unsubscribed”
clientid消息目的 Client ID
username消息目的用户名
peerhost客户端的 IPAddress
topicMQTT 主题
qosMQTT 消息的 QoS
timestamp事件触发时间 (ms)
node事件触发所在节点

SQL 关键字和符号

SELECT - FROM - WHERE 语句

SELECT 语句用于决定最终的输出结果里的字段。比如:

下面 SQL 的输出结果中将只有两个字段 “a” 和 “b”:

  1. SELECT a, b FROM "t/#"

WHERE 语句用于对本事件中可用字段,或 SELECT 语句中定义的字段进行条件过滤。比如:

  1. # 选取 username 为 'abc' 的终端发来的消息,输出结果为所有可用字段:
  2. SELECT * FROM "#" WHERE username = 'abc'
  3. ## 选取 clientid 为 'abc' 的终端发来的消息,输出结果将只有 cid 一个字段。
  4. ## 注意 cid 变量是在 SELECT 语句中定义的,故可在 WHERE 语句中使用:
  5. SELECT clientid as cid FROM "#" WHERE cid = 'abc'
  6. ## 选取 username 为 'abc' 的终端发来的消息,输出结果将只有 cid 一个字段。
  7. ## 注意虽然 SELECT 语句中只选取了 cid 一个字段,所有消息发布事件中的可用字段 (比如 clientid, username 等) 仍然可以在 WHERE 语句中使用:
  8. SELECT clientid as cid FROM "#" WHERE username = 'abc'
  9. ## 但下面这个 SQL 语句就不能工作了,因为变量 xyz 既不是消息发布事件中的可用字段,又没有在 SELECT 语句中定义:
  10. SELECT clientid as cid FROM "#" WHERE xyz = 'abc'

FROM 语句用于选择事件来源。如果是消息发布则填写消息的主题,如果是事件则填写对应的事件主题。

运算符号

函数名函数作用返回值
+加法,或字符串拼接加和,或拼接之后的字符串
-减法差值
*乘法乘积
/除法商值
div整数除法整数商值
mod取模
=比较两者是否完全相等。可用于比较变量和主题true/false
=~比较主题(topic)是否能够匹配到主题过滤器(topic filter)。只能用于主题匹配true/false

SQL 语句中可用的函数

数学函数

函数名函数作用参数返回值
abs绝对值
  1. 被操作数
绝对值
cos余弦
  1. 被操作数
余弦值
cosh双曲余弦
  1. 被操作数
双曲余弦值
acos反余弦
  1. 被操作数
反余弦值
acosh反双曲余弦
  1. 被操作数
反双曲余弦值
sin正弦
  1. 被操作数
正弦值
sinh双曲正弦
  1. 被操作数
双曲正弦值
asin反正弦
  1. 被操作数
asinh反双曲正弦
  1. 被操作数
反双曲正弦值
tan正切
  1. 被操作数
正切值
tanh双曲正切
  1. 被操作数
双曲正切值
atan反正切
  1. 被操作数
反正切值
atanh反双曲正切
  1. 被操作数
反双曲正切值
ceil上取整
  1. 被操作数
整数值
floor下取整
  1. 被操作数
整数值
round四舍五入
  1. 被操作数
整数值
exp幂运算
  1. 被操作数
e 的 x 次幂
power指数运算
  1. 左操作数 x 2. 右操作数 y
x 的 y 次方
sqrt平方根运算
  1. 被操作数
平方根
fmod负点数取模函数
  1. 左操作数 2. 右操作数
log以 e 为底对数
  1. 被操作数
log10以 10 为底对数
  1. 被操作数
log2以 2 为底对数
  1. 被操作数

数据类型判断函数

函数名函数作用参数返回值
is_null判断变量是否为空值
  1. Data
Boolean 类型的数据。如果为空值(undefined) 则返回 true,否则返回 false
is_not_null判断变量是否不为空值
  1. Data
Boolean 类型的数据。如果为空值(undefined) 则返回 true,否则返回 false
is_str判断变量是否为 String 类型
  1. Data
Boolean 类型的数据。
is_bool判断变量是否为 Boolean 类型
  1. Data
Boolean 类型的数据。
is_int判断变量是否为 Integer 类型
  1. Data
Boolean 类型的数据。
is_float判断变量是否为 Float 类型
  1. Data
Boolean 类型的数据。
is_num判断变量是否为数字类型,包括 Integer 和 Float 类型
  1. Data
Boolean 类型的数据。
is_map判断变量是否为 Map 类型
  1. Data
Boolean 类型的数据。
is_array判断变量是否为 Array 类型
  1. Data
Boolean 类型的数据。

数据类型转换函数

函数名函数作用参数返回值
str将数据转换为 String 类型
  1. Data
String 类型的数据。无法转换将会导致 SQL 匹配失败
str_utf8将数据转换为 UTF-8 String 类型
  1. Data
UTF-8 String 类型的数据。无法转换将会导致 SQL 匹配失败
bool将数据转换为 Boolean 类型
  1. Data
Boolean 类型的数据。无法转换将会导致 SQL 匹配失败
int将数据转换为整数类型
  1. Data
整数类型的数据。无法转换将会导致 SQL 匹配失败
float将数据转换为浮点型类型
  1. Data
浮点型类型的数据。无法转换将会导致 SQL 匹配失败
map将数据转换为 Map 类型
  1. Data
Map 类型的数据。无法转换将会导致 SQL 匹配失败

字符串函数

函数名函数作用参数返回值举例
lower转为小写1. 原字符串小写字符串1. lower(‘AbC’) = ‘abc’

2. lower(‘abc’) = ‘abc’
upper转为大写1. 原字符串大写字符串1. upper(‘AbC’) = ‘ABC’

2. lower(‘ABC’) = ‘ABC’
trim去掉左右空格1. 原字符串去掉空格后的字符串1. trim(‘ hello ‘) = ‘hello’
ltrim去掉左空格1. 原字符串去掉空格后的字符串1. ltrim(‘ hello ‘) = ‘hello ‘
rtrim去掉右空格1. 原字符串去掉空格后的字符串1. rtrim(‘ hello ‘) = ‘ hello’
reverse字符串反转1. 原字符串翻转后的字符串1. reverse(‘hello’) = ‘olleh’
strlen取字符串长度1. 原字符串整数值,字符长度1. strlen(‘hello’) = 5
substr取字符的子串1. 原字符串 2. 起始位置. 注意: 下标从 0 开始子串1. substr(‘abcdef’, 2) = ‘cdef’
substr取字符的子串1. 原字符串 2. 起始位置 3. 要取出的子串长度. 注意: 下标从 0 开始子串1. substr(‘abcdef’, 2, 3) = ‘cde’
split字符串分割1. 原字符串 2. 分割符子串分割后的字符串数组1. split(‘a/b/ c’, ‘/‘) = [‘a’, ‘b’, ‘ c’]
split字符串分割, 只查找左边第一个分隔符1. 原字符串 2. 分割符子串 3. ‘leading’分割后的字符串数组1. split(‘a/b/ c’, ‘/‘, ‘leading’) = [‘a’, ‘b/ c’]
split字符串分割, 只查找右边第一个分隔符1. 原字符串 2. 分割符子串 3. ‘trailing’分割后的字符串数组1. split(‘a/b/ c’, ‘/‘, ‘trailing’) = [‘a/b’, ‘ c’]
concat字符串拼接1. 左字符串 2. 右符子串拼接后的字符串1. concat(‘a’, ‘/bc’) = ‘a/bc’

2. ‘a’ + ‘/bc’ = ‘a/bc’
tokens字符串分解(按照指定字符串符分解)1. 输入字符串 2. 分割符或字符串分解后的字符串数组1. tokens(‘ a/b/ c’, ‘/‘) = [‘ a’, ‘b’, ‘ c’]

2. tokens(‘ a/b/ c’, ‘/ ‘) = [‘a’, ‘b’, ‘c’]

3. tokens(‘ a/b/ c\n’, ‘/ ‘) = [‘a’, ‘b’, ‘c\n’]
tokens字符串分解(按照指定字符串和换行符分解)1. 输入字符串 2. 分割符或字符串 3. ‘nocrlf’分解后的字符串数组1. tokens(‘ a/b/ c\n’, ‘/ ‘, ‘nocrlf’) = [‘a’, ‘b’, ‘c’]

2. tokens(‘ a/b/ c\r\n’, ‘/ ‘, ‘nocrlf’) = [‘a’, ‘b’, ‘c’]
sprintf字符串格式化, 格式字符串的用法详见 https://erlang.org/doc/man/io.html#fwrite-1 里的 Format 部分1. 格式字符串 2,3,4… 参数列表。参数个数不定分解后的字符串数组1. sprintf(‘hello, ~s!’, ‘steve’) = ‘hello, steve!’

2. sprintf(‘count: ~p~n’, 100) = ‘count: 100\n’
pad字符串补足长度,补空格,从尾部补足1. 原字符串 2. 字符总长度补足后的字符串1. pad(‘abc’, 5) = ‘abc ‘
pad字符串补足长度,补空格,从尾部补足1. 原字符串 2. 字符总长度 3. ‘trailing’补足后的字符串1. pad(‘abc’, 5, ‘trailing’) = ‘abc ‘
pad字符串补足长度,补空格,从两边补足1. 原字符串 2. 字符总长度 3. ‘both’补足后的字符串1. pad(‘abc’, 5, ‘both’) = ‘ abc ‘
pad字符串补足长度,补空格,从头部补足1. 原字符串 2. 字符总长度 3. ‘leading’补足后的字符串1. pad(‘abc’, 5, ‘leading’) = ‘ abc’
pad字符串补足长度,补指定字符,从尾部补足1. 原字符串 2. 字符总长度 3. ‘trailing’ 4. 指定用于补足的字符补足后的字符串1. pad(‘abc’, 5, ‘trailing’, ‘‘) = ‘abc**’

2. pad(‘abc’, 5, ‘trailing’, ‘
#’) = ‘abc##’
pad字符串补足长度,补指定字符,从两边补足1. 原字符串 2. 字符总长度 3. ‘both’ 4. 指定用于补足的字符补足后的字符串1. pad(‘abc’, 5, ‘both’, ‘‘) = ‘abc

2. pad(‘abc’, 5, ‘both’, ‘
#’) = ‘#abc#’
pad字符串补足长度,补指定字符,从头部补足1. 原字符串 2. 字符总长度 3. ‘leading’ 4. 指定用于补足的字符补足后的字符串1. pad(‘abc’, 5, ‘leading’, ‘‘) = ‘**abc’

2. pad(‘abc’, 5, ‘leading’, ‘
#’) = ‘##abc’
replace替换字符串中的某子串,查找所有匹配子串替换1. 原字符串 2. 要被替换的子串 3. 指定用于替换的字符串替换后的字符串1. replace(‘ababef’, ‘ab’, ‘cd’) = ‘cdcdef’
replace替换字符串中的某子串,查找所有匹配子串替换1. 原字符串 2. 要被替换的子串 3. 指定用于替换的字符串 4. ‘all’替换后的字符串1. replace(‘ababef’, ‘ab’, ‘cd’, ‘all’) = ‘cdcdef’
replace替换字符串中的某子串,从尾部查找第一个匹配子串替换1. 原字符串 2. 要被替换的子串 3. 指定用于替换的字符串 4. ‘trailing’替换后的字符串1. replace(‘ababef’, ‘ab’, ‘cd’, ‘trailing’) = ‘abcdef’
replace替换字符串中的某子串,从头部查找第一个匹配子串替换1. 原字符串 2. 要被替换的子串 3. 指定用于替换的字符串 4. ‘leading’替换后的字符串1. replace(‘ababef’, ‘ab’, ‘cd’, ‘leading’) = ‘cdabef’
regex_match判断字符串是否与某正则表达式匹配1. 原字符串 2. 正则表达式true 或 false1. regex_match(‘abc123’, ‘[a-zA-Z1-9]*’) = true
regex_replace替换字符串中匹配到某正则表达式的子串1. 原字符串 2. 正则表达式 3. 指定用于替换的字符串替换后的字符串1. regex_replace(‘ab1cd3ef’, ‘[1-9]’, ‘[&]’) = ‘ab[1]cd[3]ef’

2. regex_replace(‘ccefacef’, ‘c+’, ‘:’) = ‘:efa:ef’
ascii返回字符对应的 ASCII 码1. 字符整数值,字符对应的 ASCII 码1. ascii(‘a’) = 97
find查找并返回字符串中的某个子串,从头部查找1. 原字符串 2. 要查找的子串查抄到的子串,如找不到则返回空字符串1. find(‘eeabcabcee’, ‘abc’) = ‘abcabcee’
find查找并返回字符串中的某个子串,从头部查找1. 原字符串 2. 要查找的子串 3. ‘leading’查抄到的子串,如找不到则返回空字符串1. find(‘eeabcabcee’, ‘abc’, ‘leading’) = ‘abcabcee’
find查找并返回字符串中的某个子串,从尾部查找1. 原字符串 2. 要查找的子串 3. ‘trailing’查抄到的子串,如找不到则返回空字符串1. find(‘eeabcabcee’, ‘abc’, ‘trailing’) = ‘abcee’

Map 函数

函数名函数作用参数返回值
map_get取 Map 中某个 Key 的值,如果没有则返回空值
  1. Key 2. Map
Map 中某个 Key 的值。支持嵌套的 Key,比如 “a.b.c”
map_get取 Map 中某个 Key 的值,如果没有则返回指定默认值
  1. Key 2. Map 3. Default Value
Map 中某个 Key 的值。支持嵌套的 Key,比如 “a.b.c”
map_put向 Map 中插入值
  1. Key 2. Value 3. Map
插入后的 Map。支持嵌套的 Key,比如 “a.b.c”

数组函数

函数名函数作用参数返回值
nth取第 n 个元素,下标从 1 开始
  1. 原数组
第 n 个元素
length获取数组的长度
  1. 原数组
数组长度
sublist取从第一个元素开始、长度为 len 的子数组。下标从 1 开始
  1. 长度 len 2. 原数组
子数组
sublist取从第 n 个元素开始、长度为 len 的子数组。下标从 1 开始
  1. 起始位置 n 2. 长度 len 3. 原数组
子数组
first取第 1 个元素。下标从 1 开始
  1. 原数组
第 1 个元素
last取最后一个元素。
  1. 原数组
最后一个元素
contains判断数据是否在数组里面
  1. 数据 2. 原数组
Boolean 值

哈希函数

函数名函数作用参数返回值
md5求 MD5 值
  1. 数据
MD5 值
sha求 SHA 值
  1. 数据
SHA 值
sha256求 SHA256 值
  1. 数据
SHA256 值

编解码函数

函数名函数作用参数返回值
base64_encodeBASE64 编码
  1. 数据
BASE64 字符串
base64_decodeBASE64 解码
  1. BASE64 字符串
数据
json_encodeJSON 编码
  1. JSON 字符串
内部 Map
json_decodeJSON 解码
  1. 内部 Map
JSON 字符串
schema_encodeSchema 编码
  1. Schema ID 2. 内部 Map
数据
schema_encodeSchema 编码
  1. Schema ID 2. 内部 Map 3. Protobuf Message 名
数据
schema_decodeSchema 解码
  1. Schema ID 2. 数据
内部 Map
schema_decodeSchema 解码
  1. Schema ID 2. 数据 3. Protobuf Message 名
内部 Map

在 Dashboard 中测试 SQL 语句

Dashboard 界面提供了 SQL 语句测试功能,通过给定的 SQL 语句和事件参数,展示 SQL 测试结果。

  1. 在创建规则界面,输入 规则SQL,并启用 SQL 测试 开关:

    image

  2. 修改模拟事件的字段,或者使用默认的配置,点击 测试 按钮:

    image

  3. SQL 处理后的结果将在 测试输出 文本框里展示:

    image