从 v4 迁移到 v5

本章节提供了从 EMQX 4.x 版本迁移至 5.0.0 版本的指南,尽管 EMQX 5.0 不保证向后兼容(Backward Compatibility),但绝大部分功能运行机制没有太大的变化,你仍然可以通过手动迁移的方式来实现升级。

日志

日志输出格式进行了调整,请参照日志

默认监听器

默认将不再提供以下监听器:

监听器说明
MQTT-TCP 11883后端应用使用的 MQTT 接入点
Management-HTTP 8081REST API 端口,已合并至 18083 端口

插件

此前的官方插件已迁移到 EMQX 中成为内置功能,用户在 4.x 版本上开发的插件需要重新适配。以下是官方插件与现有功能对照表:

4.x5.0
emqx_auth_http认证/授权 - HTTP 数据源
emqx_auth_jwt认证 - JWT
emqx_auth_mnesia认证/授权 - 内置数据库
emqx_auth_mongo认证/授权 - MongoDB 数据源
emqx_auth_mysql认证/授权 - MySQL 数据源
emqx_auth_pgsql认证/授权 - PostgreSQL 数据源
emqx_auth_redis认证/授权 - Redis 数据源
emqx_sasl认证/授权 - 增强认证
emqx_auth_ldap-
emqx_rule_engine数据集成功能
emqx_bridge_mqtt数据桥接 - MQTT Sink/MQTT Source
emqx_web_hook数据桥接 - WebHook
emqx_coapCoAP 网关
emqx_dashboardDasboard
emqx_exhookExHook 功能
emqx_exprotoExProto 网关
emqx_lwm2mLwM2M 网关
emqx_snMQTT-SN 网关
emqx_stompSTOMP 网关
emqx_lua_hook-
emqx_management-
emqx_prometheusPrometheus 功能
emqx_psk_file认证 PSK (psk_authentication.enable = true)
emqx_recon通过 emqx_ctl observer 命令提供
emqx_retainerRetain 功能
emqx_telemetry遥测功能

HTTP API

此前“应用”(Appication)用于管理 API 访问凭证,现已更名为“API 密钥”(API Key),且 Secret Key 仅在创建成功时返回一次后续无法再次获得。

8081 端口已被合并至 18083 端口,API 访问基础路径由 /api/v4 切换到 /api/v5,请通过此端口和路径调用 API;时间相关的数据将使用 RFC3339从 v4 迁移到 v5 - 图1 (opens new window) 格式。

响应数据格式调整

成功响应时业务状态码 code 将不再随数据返回,出错时将返回对应的 4xx/5xx HTTP 状态码和错误提示, 用户可以访问 GET /error_codes 获取所有可能的错误码,以下是响应格式的对照示例:

  1. # 4.x 成功响应
  2. ## HTTP StatusCode = 200
  3. GET /rules/my_rule
  4. { "code": 0, "data": { ... } }
  5. # 4.x 错误响应
  6. ## HTTP StatusCode = 200
  7. GET /rules/my_rule
  8. { "code": 404, "message": "Not Found" }
  9. # 5.0 成功响应
  10. ## HTTP StatusCode = 200
  11. GET /rules/my_rule
  12. { ... }
  13. # 5.0 错误响应
  14. ## HTTP StatusCode = 404
  15. GET /rules/my_rule
  16. { "code": "NOT_FOUND", "message": "Rule Id Not Found" }

主要 API 变动

API 有较大变动,以下是常用的 API 变动对照表,现存的部分 API 我们做了兼容处理:

提示

兼容性说明:

  • 兼容:沿用了旧版 API 路径跟参数,或者保留了旧版 API
  • 部分兼容:API 路径不变,但部分 API 字段发生改变
  • 不兼容:API 路径与字段都发生变化
4.x5.0兼容性说明
发布/订阅操作
POST /mqtt/publishPOST /publish兼容
POST /mqtt/publish_batchPOST /publish/bulk兼容
POST /mqtt/subscribePOST /clients/{clientid}/subscribe兼容
POST /mqtt/subscribe_batchPOST /clients/{clientid}/subscribe/bulk兼容
POST /mqtt/unsubscribePOST /clients/{clientid}/unsubscribe兼容
POST /mqtt/unsubscribe_batchPOST /clients/{clientid}/unsubscribe/bulk兼容
连接/主题/订阅
GET /clientsGET /clients部分兼容
GET /routes{/topic}GET /topics{/topic}不兼容routes 更名为 topics
GET /subscriptionsGET /subscriptions部分兼容
GET /subscriptions/{clientid}GET /clients/{clientid}/subscriptions不兼容
节点/状态/指标
GET /nodesGET /nodes部分兼容
GET /brokers-不兼容合并至 GET /nodes
GET /statsGET /stats部分兼容
GET /metricsGET /metrics部分兼容
用户/告警
GET /usersGET /users部分兼容
GET /alarms{/activated}GET /alarms?activated={true,false}不兼容
GET /alarms{/deactivated}GET /alarms?activated={true,false}不兼容

认证/发布订阅 ACL

认证授权插件(emqx_auth_*)已被移除,相关能力以内置功能的形式迁移到 EMQX 中,支持用户以 Dashboard 或配置文件的方式配置。

发布订阅 ACL 更名为 授权,认证与发布订阅 ACL 功能进行了拆分。

我们保留了 4.x 中的认证方式与支持的数据源,但使用方式上有一定调整。

固定的执行顺序

同时启用多个认证器或授权检查器时,不再按照启动顺序而是按照固定的配置顺序来执行检查,可在配置文件跟 Dashboard 中调整的执行顺序。

更换占位符变量提取语法

此前认证插件中可以使用 %u 类似的语法构造占位符以提取变量,将客户端信息动态拼接到如 SQL 语句、Redis 查询命令和 HTTP 请求中, 现在 EMQX 所有认证跟授权检查功能中使用了新的语法 ${},如 ${username}${clientid},该语法与规则 SQL 一致。

以下是前后版本配置对比:

  1. # 4.x
  2. # etc/emqx_auth_mysql.conf
  3. auth.mysql.auth_query = select password from mqtt_user where username = '%u' limit 1
  4. # 5.0
  5. # emqx.conf
  6. authentication = [
  7. {
  8. ...
  9. mechanism = "password_based"
  10. backend = "mysql"
  11. query = "SELECT password_hash, salt FROM mqtt_user where username = ${username} LIMIT 1"
  12. }
  13. ]

认证

移除匿名认证机制

移除 allow_anonymous 配置项,默认允许所有客户端连接,添加并启用任意一个认证器后将对所有新连接进行认证检查。

移除 bypass_auth_plugins 配置项,某个监听器需要禁用认证时,可以通过 listeners.{type}.{name}.enable_authn = true | false 配置项进行设置。

内置数据库 (Mnesia) 变动

  1. 为方便理解,该数据源由 Mnesia 更名为内置数据库(Built-in Database);
  2. 只能选定一种认证查找方式:基于用户名或基于客户端 ID;
  3. 数据格式与数据操作 REST API 有变动,详见 POST /authentication/{id}/users

用户可以使用数据导入 API 将旧版本中的数据导入到 EMQX 5.0 中,详见 POST /authentication/{id}/import_users

HTTP 变动

  1. 使用响应 Body 里面的 JSON 字段而非响应状态码(HTTP response status codes)判断认证结果;
  2. 移除超级用户(superuser)查询,如需超级用户功能请在响应 body 里面通过 JSON 设置。

成功响应状态码:

  1. 200 204

其他状态码或请求失败则忽略该认证器。

成功响应 Body (JSON):

名称类型必选说明
result枚举trueallow | deny | ignore
is_supseruser布尔false默认为 false
  1. {
  2. "result": "allow",
  3. "is_supseruser": true
  4. }

MySQL/PostgreSQL 变动

  1. 查询结果中要求的密码字段由 password 更改为 password_hash,在不更改数据库列名的情况下可以在查询时使用 as 语法完成迁移;
  2. 移除超级用户(superuser)查询 SQL,如需超级用户功能请确保认证 SQL 结果包含 is_superuser 字段。
  1. SELECT password as password_hash, salt, is_superuser FROM mqtt_user where username = ${username} LIMIT 1

MongoDB 变动

  1. 移除超级用户(superuser)查询,如需超级用户功能请在 MongoDB 认证器配置指定 is_superuser_field 字段,并确保认证查询结果中包含超级用户信息。
  1. authentication = [
  2. {
  3. ...
  4. mechanism = "password_based"
  5. backend = "mongodb"
  6. # is_superuser_field = "is_superuser"
  7. }
  8. ]

Redis 变动

  1. 仅支持 Redis Hashes从 v4 迁移到 v5 - 图2 (opens new window) 数据结构与 HGETHMGET 查询命令,必须使用 password_hashpassword(兼容 4.x)作为密码字段名;
  2. 移除超级用户(superuser)查询,如需超级用户功能请在 Redis 查询命令中添加 is_superuser 字段。
  1. # bad
  2. GET emqx_user:${username}
  3. # bad
  4. HMGET emqx_user:${username} passwd
  5. # good
  6. HMGET emqx_user:${username} password_hash
  7. # good
  8. HMGET emqx_user:${username} password_hash is_superuser

LDAP 变动

暂不支持 LDAP 认证。

JWT 变动

无变动。

授权(ACL)

ACL File 变动

  1. 移除 acl_file 配置,基于文件的 ACL(acl.conf) 将作为授权检查器中的一种,默认添加到 EMQX 中;
  2. acl.conf 文件从 etc 目录移动到 data/authz 目录,请确保 EMQX 具有该文件的读写权限;
  3. acl.conf 数据文件语法有所调整,变更情况见下表:
4.x5.0.0是否兼容
userusername
clientclientid
pubsuball

内置数据库 (Mnesia) 变动

  1. 为方便理解,该数据源由 Mnesia 更名为内置数据库(Built-in Database);
  2. 数据格式与数据操作 REST API 有变动,详见 POST /authorization/built_in_database/clientid

4.x 中的数据导出命令 ./bin/emqx_ctl data export 导出数据中包含 ACL 数据,用户可以将数据处理为 5.0 所需格式通过对应的 REST API 进行导入。

MySQL/PostgreSQL 变动

  1. 查询结果中不再需要 ipaddr/username/clientid 字段;
  2. access 字段名变为 action,数据类型由整型变为字符或字符枚举,数值对应关系见下表;
  3. allow 字段名变为 permission,数据类型由整型变为字符或字符枚举,数值对应关系见下表。

access-action 字段数据类型映射

4.x (int)5.0.0 (varchar/enum)对应动作
1subscribe订阅
2publish发布
3all发布订阅

allow-permission 字段数据类型映射

4.x (int)5.0.0 (varchar/enum)对应行为
0deny拒绝
1allow允许

MongoDB 变动

  1. MongoDB 数据源可用于黑/白名单模式下,此前仅支持白名单模式,要求设置 acl_nomatch = deny
  2. 需要从 MongoDB 中查询出包含 action permission topics 字段的数据列表,使用方式详见 AuthZ-MongoDB

如需继续使用 4.x 中的数据,请手动迁移适配。

5.0 中数据示例

  1. [
  2. {
  3. "username": "emqx_u",
  4. "clientid": "emqx_c",
  5. "ipaddress": "127.0.0.1",
  6. "permission": "allow",
  7. "action": "all",
  8. "topics": ["#"]
  9. }
  10. ]

Redis 变动

  1. Redis 数据源仍然仅支持白名单模式,要求设置 acl_nomatch = deny
  2. access 字段名变为 action,数据由数字变为动作全称字符串,对应关系见下表。
4.x5.0.0对应动作
1subscribe订阅
2publish发布
3all发布订阅

如需继续使用 4.x 中的数据,请手动迁移适配。

5.0 中数据示例

  1. HSET mqtt_acl:emqx_u t/# subscribe
  2. HSET mqtt_acl:emqx_u # all
  3. HSET mqtt_acl:emqx_u a/1 publish

HTTP 变动

  1. 使用响应 Body 里面的 JSON 字段而非响应状态码(HTTP response status codes)判断认证结果。

成功响应状态码:

  1. 200 204

其他状态码或请求失败则忽略该授权检查器。

成功响应 Body (JSON):

名称类型必选说明
result枚举trueallow | deny | ignore
  1. {
  2. "result": "deny"
  3. }

规则引擎

规则引擎已更名为数据集成,包含规则与数据桥接功能。

规则 SQL 完全兼容 4.x 的语法,但规则下的动作拆分为内置动作(republish、console)与数据桥接(WebHook、MQTT Sink、MQTT Source),以便实现”动作”的复用。

WebHook

WebHook 插件(emqx_web_hook) 已被移除,请使用数据集成中的 WebHook 数据桥接功能替代。

MQTT 桥接

MQTT 桥接插件(emqx_bridge_mqtt) 已被移除,请使用数据集成中的 MQTT Sink 和 MQTT Source 数据桥接功能替代。

Prometheus

旧的 Prometheus 插件 (emqx_prometheus) 已被移除。 在5.0中,Prometheus 的数据拉取服务默认开启,且无需认证就可以拉取数据。 可以使用 curl 命令来查看:curl -f “127.0.0.1:18083/api/v5/prometheus/stats”

如果您想要使用 push-gateway,您可以在 prometheus {…} 配置块中启用它。 或者在仪表盘中修改配置并启用。

指标变动情况:

4.4.45.0说明
emqx_client_auth_success_anonymousemqx_client_auth_anonymous更名
emqx_client_check_aclemqx_client_authorize counter更名
-emqx_mria_last_intercepted_trans新增
-emqx_mria_replicants新增
-emqx_mria_server_mql新增
-emqx_mria_weight新增
emqx_routes_countemqx_topics_count更名
emqx_routes_maxemqx_topics_max更名
emqx_session_takeoveredemqx_session_takenover更名
erlang_vm_ets_tables-移除
-erlang_vm_memory_dets_tables新增
-erlang_vm_memory_ets_tables新增
-erlang_vm_msacc_alloc_seconds_total新增
-erlang_vm_msacc_aux_seconds_total新增
-erlang_vm_msacc_bif_seconds_total新增
-erlang_vm_msacc_busy_wait_seconds_total新增
-erlang_vm_msacc_check_io_seconds_total新增
-erlang_vm_msacc_emulator_seconds_total新增
-erlang_vm_msacc_ets_seconds_total新增
-erlang_vm_msacc_gc_full_seconds_total新增
-erlang_vm_msacc_gc_seconds_total新增
-erlang_vm_msacc_nif_seconds_total新增
-erlang_vm_msacc_other_seconds_total新增
-erlang_vm_msacc_port_seconds_total新增
-erlang_vm_msacc_send_seconds_total新增
-erlang_vm_msacc_sleep_seconds_total新增
-erlang_vm_msacc_timers_seconds_total新增
-erlang_vm_statistics_dirty_cpu_run_queue_length新增
-erlang_vm_statistics_dirty_io_run_queue_length新增
-erlang_vm_wordsize_bytes新增

网关/多协议接入

其他协议(LwM2M、CoAP、STOMP、MQTT-SN)的客户端将不再映射为 MQTT 客户端,无法通过 Dashboard 客户端页面和 GET /clients API 获取。 用户可以前往网关页面详情页面或通过 GET /gateway/{name}/clients API 获取。