📣 极限科技诚招搜索运维工程师(Elasticsearch/Easysearch)- 全职/北京 👉 : 立即申请加入

适用版本: 6.8-8.9

1. 错误异常的基本描述 #

could not parse message attachment. failed to parse [...] field 是 Elasticsearch Watcher 在解析消息附件(message attachment)时抛出的异常。当 Watcher 尝试解析附件中的模板字段(如 fallbackcolor 等)时,如果这些字段的值不符合 TextTemplate 的解析规则,就会触发此错误。

该错误常见于 Slack、邮件等 action 的 attachment 配置中,属于字段级别的解析失败,而非整个 attachment 结构错误。

常见现象 #

  • 创建或更新 watch 时返回 HTTP 400 错误。
  • Watcher 执行时,包含 attachment 的 action 失败。
  • 报错信息中会明确指出是哪个字段解析失败(如 failed to parse [fallback] field)。
  • 常见于手动编写复杂的模板表达式时,或在使用 Mustache 模板语法时出错。

典型报错与异常栈 #

{
  "error": {
    "root_cause": [
      {
        "type": "parse_exception",
        "reason": "could not parse message attachment. failed to parse [fallback] field"
      }
    ],
    "type": "parse_exception",
    "reason": "could not parse message attachment. failed to parse [fallback] field"
  },
  "status": 400
}

服务端日志中可能出现类似以下内容:

[2024-01-15T10:30:00,123][WARN ][o.e.x.w.a.e.SlackAction ] [node-1] failed to parse message attachment
ElasticsearchParseException[could not parse message attachment. failed to parse [fallback] field]
    at org.elasticsearch.xpack.watcher.actions.email.EmailAttachmentParser.parse(EmailAttachmentParser.java:89)
    at org.elasticsearch.xpack.watcher.actions.email.EmailAction.execute(EmailAction.java:123)
Caused by: ElasticsearchParseException[failed to parse [fallback] field]

2. 为什么会发生这个错误 #

Watcher 的 message attachment 中某些字段(如 fallbackcolortexttitle 等)需要使用 TextTemplate 格式进行解析。当这些字段的值结构不符合 TextTemplate 的解析规则时,就会抛出此异常。

常见原因包括:

  • 模板字段值结构错误TextTemplate 期望的值是字符串或包含 inlinescript 等特定结构的对象,如果传入了不兼容的结构(如数组、嵌套对象等),就会解析失败。
  • 类型错误:本应是模板/字符串的字段被写成了数组或非法对象。例如,将 fallback 字段设置为数组而非字符串。
  • 模板变量渲染异常:如果使用了动态模板变量(如 {{ctx.payload.result}}),变量渲染后的结果可能不是期望的类型。
  • Mustache 模板语法错误:在模板字符串中使用了不正确的 Mustache 语法,导致解析失败。
  • 字段值为 null 或空对象:某些字段不允许为 null 或空对象,但配置中提供了这样的值。
  • JSON 结构嵌套错误:attachment 的 JSON 结构嵌套层级不正确,导致解析器无法正确识别字段类型。

3. 如何排查和解决这个异常和解决这个异常 #

排查步骤 #

  1. 确认报错中的字段名:从错误信息中提取具体是哪个字段解析失败(如 fallbackcolortext 等)。

  2. 检查 Watch 中对应字段的配置

# 获取 watch 配置
GET _watcher/watch/<watch_id>?pretty

# 检查 attachment 部分配置
  1. 验证字段值的类型:确认该字段的值是否符合 TextTemplate 格式要求。
# TextTemplate 合法格式示例
# 1. 简单字符串
"fallback": "This is a fallback text"

# 2. 内联模板
"fallback": {
  "inline": "Alert: {{ctx.payload.hits.total}} hits"
}

# 3. 脚本模板
"fallback": {
  "script": {
    "source": "return 'Alert: ' + ctx.payload.hits.total"
  }
}
  1. 使用简化配置测试:将复杂模板替换为简单字符串,确认是否是模板结构问题。
# 先用简单字符串测试
PUT _watcher/watch/<watch_id>
{
  "actions": {
    "send_slack": {
      "slack": {
        "message": {
          "attachments": [
            {
              "fallback": "Simple fallback text",
              "color": "#ff0000",
              "text": "Test message"
            }
          ]
        }
      }
    }
  }
}

排查时需要注意的问题 #

  • 注意区分 TextTemplate 的两种格式:简单字符串格式和对象格式(包含 inlinescript 字段)。
  • 如果使用了脚本模板,检查脚本语法是否正确,以及脚本返回值的类型是否符合字段期望。
  • 检查是否有 Mustache 模板变量在渲染后产生了意外的类型转换。

4. 如何解决这个错误 #

常用修复思路 #

  1. 修正字段格式:确保 attachment 字段使用正确的格式。
# 正确的 fallback 字段格式
PUT _watcher/watch/<watch_id>
{
  "actions": {
    "send_email": {
      "email": {
        "attachments": [
          {
            "fallback": "Alert notification",
            "color": "danger",
            "title": "Elasticsearch Alert",
            "text": "{{ctx.payload.hits.total}} documents matched"
          }
        ]
      }
    }
  }
}
  1. 使用合法的颜色值color 字段必须是合法的颜色名称(如 goodwarningdanger)或十六进制颜色码(如 #ff0000)。

  2. 简化模板表达式:如果模板过于复杂,考虑拆分成多个简单字段,或使用 script 模板替代 Mustache 模板。

  3. 添加字段值校验:在更新 watch 前,使用 JSON Schema 或类似工具校验 attachment 结构。

后续注意事项与推荐建议 #

  • 在测试环境先验证 attachment 配置,特别是包含复杂模板的场景。
  • 建立 attachment 配置的模板库,复用经过验证的配置片段。
  • 对动态模板变量进行类型检查,确保渲染后的结果符合字段类型要求。

借助 INFINI 产品提升排障效率 #

  • INFINI Console 提供可视化的 Watch 管理和测试功能,可以实时预览模板渲染结果,帮助快速发现模板字段配置问题。
  • INFINI Gateway 可以记录 Watcher action 的完整请求和响应内容,帮助捕获 attachment 解析失败的详细上下文,并通过流量回放验证修复方案。
  • 通过 INFINI Console 的模板测试功能,可以在不执行完整 Watch 的情况下验证模板字段的正确性。

5. 小结 #

could not parse message attachment. failed to parse [...] field 是一个字段级别的解析错误,通常由于 attachment 中某个模板字段(fallbackcolor 等)的值不符合 TextTemplate 格式要求导致。解决此问题的关键是:确认报错字段名,检查该字段的值类型和结构,修正为符合 TextTemplate 格式的合法值。

通过 INFINI Console 的模板验证功能和 INFINI Gateway 的请求捕获能力,可以更高效地定位和修复 message attachment 的配置问题。

相关错误 #

参考文档 #

附:日志上下文 #

下面保留当前页面中的源码或日志片段,便于继续结合异常调用栈定位问题:

currentFieldName = parser.currentName();
} else if (XField.FALLBACK.match(currentFieldName, parser.getDeprecationHandler())) {
    try {
        fallback = TextTemplate.parse(parser);
    } catch (ElasticsearchParseException pe) {
        throw new ElasticsearchParseException("could not parse message attachment. failed to parse [{}] field", pe,
            XField.FALLBACK);
    }