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

适用版本: 6.8-8.9

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

could not parse [schedule_type] schedule. unexpected field [field] 是 Elasticsearch Watcher 在解析 schedule(调度)配置时抛出的异常。此错误表示调度定义中出现了当前调度解析器不接受的字段名。

常见现象 #

  • 创建或更新 watch 时返回 HTTP 400 错误。
  • 报错信息明确指出哪个字段是未预期的(unexpected field [field])。
  • 常见于手动编写 schedule 配置时,或从不同版本的配置示例复制粘贴时。
  • 如果字段名拼写错误或把别的 schedule 类型字段混到当前类型里,就容易触发此错误。

典型报错与异常栈 #

{
  "error": {
    "root_cause": [
      {
        "type": "parse_exception",
        "reason": "could not parse [daily] schedule. unexpected field [invalid_field]"
      }
    ],
    "type": "parse_exception",
    "reason": "could not parse [daily] schedule. unexpected field [invalid_field]"
  },
  "status": 400
}

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

[2024-01-15T10:30:00,123][WARN ][o.e.x.w.t.s.DailySchedule] [node-1] failed to parse schedule
ElasticsearchParseException[could not parse [daily] schedule. unexpected field [invalid_field]]
    at org.elasticsearch.xpack.watcher.trigger.schedule.DailySchedule.parse(DailySchedule.java:89)
    at org.elasticsearch.xpack.watcher.trigger.schedule.Schedule.parse(Schedule.java:123)

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

Watcher 的 schedule 解析器为每种调度类型维护了一个字段白名单。当解析器遇到不在白名单中的字段时,就会抛出 unexpected field 异常。

常见原因包括:

  • schedule 字段名拼写错误:例如将 at 误写为 timeon 误写为 day 等。
  • 把别的 schedule 类型字段混到当前类型里:例如把 weeklyon 字段用到 daily schedule 里。
  • 使用旧版本示例中的遗留字段:不同版本的 Watcher 支持的字段可能不同。
  • JSON 结构嵌套错误:字段放错了层级。
  • 模板渲染问题:使用 Mustache 或 script 模板时,渲染结果可能包含不支持的字段。
  • 复制粘贴错误:从其他调度类型的配置复制字段,这些字段在当前调度类型中不支持。

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

排查步骤 #

  1. 确认报错中的调度类型和字段名:从错误信息中提取调度类型(dailyweekly 等)和未预期的字段名(unexpected field [field])。

  2. 检查 Watch 中的 schedule 配置

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

# 检查 trigger 部分的 schedule
  1. 对照官方文档验证字段合法性:确认该字段是否在当前调度类型支持的字段列表中。

  2. 检查字段层级是否正确

# 正确的 daily schedule 结构示例
PUT _watcher/watch/<watch_id>
{
  "trigger": {
    "schedule": {
      "daily": {
        "at": ["12:00", "18:00"]
      }
    }
  },
  ...
}

各调度类型支持的字段:

  • interval: 字符串值(如 "5m"
  • cron: 字符串值(cron 表达式)
  • daily: at(时间数组)
  • weekly: on(星期数组)、at(时间数组)
  • monthly: on(日期数组)、at(时间数组)
  • yearly: in(月份数组)、on(日期数组)、at(时间数组)

排查时需要注意的问题 #

  • 注意区分不同调度类型支持的字段集合:它们支持的字段不同。
  • 如果使用了模板变量,确保变量渲染后不会产生不支持的字段。
  • 检查是否有从其他调度类型复制过来的字段,这些字段在当前调度类型中可能不支持。

4. 如何解决这个错误 #

常用修复思路 #

  1. 删除非法字段:从 schedule 对象中移除不支持的字段。
# 修复前(包含不支持的字段)
PUT _watcher/watch/<watch_id>
{
  "trigger": {
    "schedule": {
      "daily": {
        "at": ["12:00"],
        "invalid_field": "some value"  # 不支持的字段
      }
    }
  }
}

# 修复后(移除不支持的字段)
PUT _watcher/watch/<watch_id>
{
  "trigger": {
    "schedule": {
      "daily": {
        "at": ["12:00"]
      }
    }
  }
}
  1. 改用正确的调度类型定义:如果字段应该属于其他调度类型,改用正确的调度类型。
# 如果 `on` 字段应该属于 weekly 而不是 daily
# 错误:在 daily 中使用 on
"daily": {
  "on": ["MON"],  # 错误
  "at": ["12:00"]
}

# 正确:改用 weekly
"weekly": {
  "on": ["MON"],  # 正确
  "at": ["12:00"]
}
  1. 修正字段拼写:检查字段名是否拼写错误。

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

  • 在编写 schedule 配置时,严格参照官方文档中的字段列表,不要随意添加字段。
  • 对不同调度类型单独维护模板,避免字段混用。
  • 在测试环境验证 schedule 配置,特别是在从其他来源复制配置时。

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

  • INFINI Console 提供可视化的 Watch 编辑功能,可以在保存前验证 schedule 配置的字段合法性,自动检测不支持的字段。
  • INFINI Gateway 可以记录 Watcher API 的完整请求内容,帮助捕获 schedule 配置的详细错误上下文,并通过流量回放验证修复方案。
  • 通过 INFINI Console 的配置对比功能,可以对比不同版本间的配置差异,快速定位版本兼容性问题。

5. 小结 #

could not parse [schedule_type] schedule. unexpected field [field] 是一个字段合法性错误,表示 schedule 中包含了不支持的字段。解决此问题的关键是:确认报错字段名,检查字段是否应该存在于当前调度类型中,移除不支持的字段或改用正确的调度类型。

通过 INFINI Console 的配置验证功能和 INFINI Gateway 的请求捕获能力,可以更高效地定位和修复 schedule 的字段层级问题。

相关错误 #

参考文档 #

附:日志上下文 #

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

} else {
    throw new ElasticsearchParseException("could not parse [{}] schedule. unexpected field [{}]", TYPE, currentFieldName);
}

return times.isEmpty() ? new DailySchedule() : new DailySchedule(times.toArray(DayTimes[]::new));