Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Rejected by Elasticsearch [error type]: document_parsing_exception [reason]: '[1:660] failed to parse field [kubernetes.labels.app] of type [text] in document with id #1041

Open
1 of 2 tasks
eli-gc opened this issue Dec 11, 2023 · 25 comments

Comments

@eli-gc
Copy link

eli-gc commented Dec 11, 2023

(check apply)

  • read the contribution guideline
  • (optional) already reported 3rd party upstream repository or mailing list if you use k8s addon or helm charts.

Problem

I cannot upgrade past 1.15.1 or else I get this error. There is no error in 1.15.0. I did not see any breaking changes in the release notes of fluentd 1.15.1

#0 dump an error event: error_class=Fluent::Plugin::ElasticsearchErrorHandler::ElasticsearchError error="400 - Rejected by Elasticsearch [error type]: document_parsing_exception [reason]: '[1:660] failed to parse field [kubernetes.labels.app] of type [text] in document with id

Steps to replicate

Either clone and modify https://gist.github.com/pitr/9a518e840db58f435911

OR

Provide example config and message

Expected Behavior or What you need to ask

fluentd.conf: |
    # set system level configurations
    <system>
      log_level debug
    </system>
    # input plugin that collects metrics from MonitorAgent
    <source>
      @type prometheus_monitor
      <labels>
        host ${hostname}
      </labels>
    </source>

    # input plugin that collects metrics for output plugin
    <source>
      @type prometheus_output_monitor
      <labels>
        host ${hostname}
      </labels>
    </source>

    # Monitors Fluentd with Datadog
    <source>
      @type monitor_agent
      bind 0.0.0.0
      port 24220
    </source>

    # input plugin to concatenate long log messages
    <filter **>
      @id containerd_concat
      @type concat
      key log
      use_first_timestamp true
      partial_key logtag
      partial_value P
      separator ""
    </filter>

    # Ignore fluentd own events
    <match fluent.**>
      @type null
    </match>


    # HTTP input for the liveness and readiness probes
    <source>
      @type http
      port 9880
    </source>

    # Throw the healthcheck to the standard output instead of forwarding it
    <match fluentd.healthcheck>
      @type stdout
    </match>

    <filter **>
      @type parser
      key_name log
      <parse>
        @type multi_format
        <pattern>
          format json
        </pattern>
        <pattern>
          format none
        </pattern>
      </parse>
    </filter>

    <source>
      @type tail
      @id in_tail_container_logs
      path /var/log/containers/*.log
      pos_file /var/log/fluentd-docker.pos
      ignore_older 1h
      tag kubernetes.*
      read_from_head true
      <parse>
        @type multi_format
        <pattern>
          # for docker cri
          format json
          time_key time
          time_format %Y-%m-%dT%H:%M:%S.%NZ
          keep_time_key true
        </pattern>
        <pattern>
          # for containerd cri
          # format /^(?<time>.+) (?<stream>stdout|stderr) [^ ]* (?<log>.*)$/
          format /^(?<time>.+) (?<stream>stdout|stderr) (?<logtag>[FP]) (?<log>.*)$/
          time_format %Y-%m-%dT%H:%M:%S.%N%:z
          keep_time_key true
        </pattern>
      </parse>
    </source>

    <filter kubernetes.var.log.containers.**.log>
      @type kubernetes_metadata
    </filter>

     # Filter for core namespace
     <filter kubernetes.var.log.containers.**.log>
       @type grep
       <exclude>
         key $.kubernetes.namespace_name
         pattern /^kube.+/
       </exclude>
     </filter>

    # Send the logs to the standard output
    <match **>
      @type elasticsearch
      log_es_400_reason true
      ssl_verify false
      reload_connections false
      reconnect_on_error true
      reload_on_failure true
      request_timeout 300s
      suppress_type_name true
      bulk_message 30
      include_tag_key true
      host "***"
      user "***"
      password "***"
      port 9243
      scheme https
      index_name "logs-aks"
      logstash_format true
      logstash_prefix "logs-aks"
      logstash_dateformat %Y%m%d
      <buffer>
        @type file
        path /opt/bitnami/fluentd/logs/buffers/logs.buffer
        flush_thread_count 1
        flush_interval 5s
        flush_mode interval
        slow_flush_log_threshold 140s
      </buffer>
    </match>

Using Fluentd and ES plugin versions

  • OS version: Debian 11.4
  • Bare Metal or within Docker or Kubernetes or others? Kubernetes
  • Fluentd v0.12 or v0.14/v1.0
    • fluentd 1.15.1
  • ES plugin 3.x.y/2.x.y or 1.x.y
    • fluent-plugin-elasticsearch (5.2.3)
  • ES version (optional)
  • 8.11.1
  • ES template(s) (optional)
@eli-gc
Copy link
Author

eli-gc commented Dec 14, 2023

I think it's related to my kubernetes metadata labels. I have two labels: app and app.kubernetes.io/name. I believe it is being rejected because one is a text type and the other is a nested object so Elasticsearch doesn't know how to handle it. Was there a change in how types or dots are handled past 1.15.0? I didn't see anything in the change log. It works just by rolling back to 1.15.0 from 1.15.1+ so I know it isn't Elasticsearch version.

@cosmo0920
Copy link
Collaborator

I think it's related to my kubernetes metadata labels. I have two labels: app and app.kubernetes.io/name. I believe it is being rejected because one is a text type and the other is a nested object so Elasticsearch doesn't know how to handle it. Was there a change in how types or dots are handled past 1.15.0?

This is what the root cause of this issue. For handling this, you need to install ES template to define the field type.
see: https://www.elastic.co/guide/en/elasticsearch/reference/8.11/mapping.html
see also: https://www.elastic.co/jp/blog/antidote-index-mapping-exceptions-ignore_malformed

@danieltaub96
Copy link

Same Issue for me, it's prevent me to upgrade to latest version, it used to work and just stopped

@eli-gc
Copy link
Author

eli-gc commented Jan 3, 2024

@cosmo0920 Is ES template a plugin? Or are you saying I need to make a template myself?

@cosmo0920
Copy link
Collaborator

cosmo0920 commented Jan 8, 2024

You guys need to create and install Elasticsearch mappings by yourself.
Auto mapping sometimes causes mistypes on their handled documents.

@eli-gc
Copy link
Author

eli-gc commented Jan 8, 2024

Thanks, I'll give it a shot and report back.

@eli-gc
Copy link
Author

eli-gc commented Jan 12, 2024

I wasn't able to get the mapping to work. It says app cannot be changed from text to ObjectMapper.

PUT /mapping-test-index
{
  "mappings": {
    "properties": {
      "app": {
        "type": "text"
      },
      "app.kubernetes.io/name": {
        "type": "text"
      }
    }
  }
}

@xdubois
Copy link

xdubois commented Jan 30, 2024

@eli-gc Did you find a working configuration ?
I've been experiencing the same conflict issue with an "app" string label

@eli-gc
Copy link
Author

eli-gc commented Jan 30, 2024

@xdubois I did not. We decided to move away from Fluentd, but you could try adding the de_dot filter manually or possibly use flattened. De_dot got removed from Fluentd which was the root of my issue. Check out these issues for more info:
de_dot removal
elasticsearch#63530

@cosmo0920
Copy link
Collaborator

Not sure the one of the solution candidates but Fluentd has dedot filter plugin: https://github.com/lunardial/fluent-plugin-dedot_filter
This should replace dot(.) with a specified character.

@xdubois
Copy link

xdubois commented Feb 7, 2024

Thanks for responses guys
Couldn't make it work with the dedot plugin
We switched to filebeat for the ease of configuration

@Rohlik
Copy link

Rohlik commented Apr 10, 2024

I have the same issue with the logging-operator version 1.6.0, which uses FluentD with this plugin. I get error below:

"reason"=>"[1:1018] failed to parse field [kubernetes.labels.app] of type [keyword] in document with id 'piLbx44BBp6m9YwiO00W'. Preview of field's value: '{kubernetes={io/component=controller}}'", "caused_by"=>{"type"=>"illegal_state_exception", "reason"=>"Can't get text on a START_OBJECT at 1:988"}

Even with dedot filter it doesn't seem to work.

  <filter **>
    @type dedot
    @id clusterflow:logging:nginx:2
    de_dot_separator _
  </filter>

@cosmo0920
Copy link
Collaborator

failed to parse field [kubernetes.labels.app] of type [keyword] in document

This is because the pointed field is not keyword i.e. just a within 256 length text. Depending on the automatic mapping caused this issue.

@Rohlik
Copy link

Rohlik commented Apr 11, 2024

@cosmo0920 Well, filebeat, which was used before on our deployment, provides label field as kubernetes.labels.app_kubernetes_io/component with its value controller, however fluent-bit/fluentd based on that log above messed up and the result is kubernetes.labels.app with its value kubernetes={io/component=controller}.
The original format in Kubernetes looks like: app.kubernetes.io/component: controller

Of course, I can tweak mapping on the Elasticsearch side, but that won't solve weird parsing.

@cosmo0920
Copy link
Collaborator

Hmm.., it's weird. Just for my curiosity, isn't it solved by using fluent-bit instead of Fluentd with this plugin?

@Rohlik
Copy link

Rohlik commented Apr 15, 2024

@cosmo0920 I deleted mapping for my index to allow all data to come in.
And I noticed that the problematic label displayed in Kibana as kubernetes.labels.app.kubernetes.io/component with its value controller.
So, I guess we can work with that and rewrite our mapping.
However, I am still wondering about that log I posted, as the error message is quite confusing because, based on that, the value should look like kubernetes={io/component=controller}.

@cosmo0920
Copy link
Collaborator

The error message came from Elasticsearch itself. So, we couldn't display more clearly unfortunately.

@xj90713
Copy link

xj90713 commented May 9, 2024

Recently I also have same problem,but I solved it.
I use the fluentd filter plugin,and the configuration information is as follows:

<filter kubernetes.**>
  @type record_transformer
  remove_keys $.docker.container_id,$.kubernetes.container_image_id,$.kubernetes.pod_id,$.kubernetes.namespace_id,$.kubernetes.master_url,$.kubernetes.labels.pod-template-hash,$['kubernetes']['labels']['app.kubernetes.io/instance'],$['kubernetes']['labels']['app.kubernetes.io/managed-by'],$['kubernetes']['labels']['app.kubernetes.io/version'],$['kubernetes']['labels']['app.kubernetes.io/name'],$['kubernetes']['labels']['app.kubernetes.io/component'],$['kubernetes']['labels']['app.kubernetes.io/part-of']
</filter>

Then,I upgrade fluentd helm package.I hope I can help you.
fluentd filter plugin

@eli-gc
Copy link
Author

eli-gc commented May 9, 2024

@xiaojun90713 so you just deleted the keys? What if I need the data from them?

@xj90713
Copy link

xj90713 commented May 9, 2024

@xiaojun90713 so you just deleted the keys? What if I need the data from them?
The logs data can still be collected normally, but I don’t need the labels for these logs. In one other hand, I have other label for these logs collect, it does not affect my collection of logs.
As a k8s log collector, I already have enough labels to query logs. For me, these remove_keys are useless. Of course, if you need to keep these labels, you can replace them with others through the filters plugin.
I don’t know if you understand what I mean.

@eli-gc
Copy link
Author

eli-gc commented May 9, 2024

@xiaojun90713 Thanks for the explanation. I suppose I could try to remove one of the labels that are colliding for me. That might work for me. However, your approach is removing the problem rather than a solution. The caveat is It only works if those labels are not required.

@fuadayodeji
Copy link

I am currently still having this issue is there anyone with any fix?

@pidumenk
Copy link

pidumenk commented Nov 6, 2024

@cosmo0920 Do you have any support of Replace_Dots option similarly to fluent-bit which comes out of the box for elasticsearch output plugin? In case you don't have, cause I haven't found it, could you add support of this feature?

At this moment many people are struggling with the following error: "error"=>{"type"=>"mapper_parsing_exception", "reason"=>"object mapping for [kubernetes.labels.app] tried to parse field [app] as object, but found a concrete value"}}}

This fix of replace field name dots with underscore is commonly used to resolve such problems.

@cosmo0920
Copy link
Collaborator

How about using https://github.com/lunardial/fluent-plugin-dedot_filter plugin? It would be great to add it on your pipeline.
We usually don't like to create monolithic plugin. So, we recommend to use it instead of adding de dotting feature in this plugin.

@Handsome-Jaxon
Copy link

Try this
Using filter with record_modifier is able to convert . to _, hope this is helpful to everyone have such issue. 😄

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

9 participants