From 120a626d628edef3ce667e3c4943c28a5e73a8b5 Mon Sep 17 00:00:00 2001 From: Leandro Beretta Date: Thu, 11 Sep 2025 10:15:41 -0300 Subject: [PATCH 1/3] Support for configure grpc client to send logs to Loki --- api/flowcollector/v1beta2/helper.go | 9 + ...observ-operator.clusterserviceversion.yaml | 6 +- .../v1.9.2-community/bundle.yaml | 84 ++ .../v1.9.2-community/index.yaml | 6 + .../v1.9.2-community/other.yaml | 7 + .../unreleased/v1.9.2-community/bundle.yaml | 840 ++++++++++++++++++ .../unreleased/v1.9.2-community/index.yaml | 6 + .../unreleased/v1.9.2-community/other.yaml | 7 + ...observ-operator.clusterserviceversion.yaml | 4 +- go.mod | 2 + go.sum | 22 +- .../consoleplugin/consoleplugin_objects.go | 22 +- .../consoleplugin/consoleplugin_test.go | 8 +- .../controller/flowcollector_controller.go | 2 +- internal/controller/flp/flp_controller.go | 2 +- .../controller/flp/flp_pipeline_builder.go | 101 ++- internal/controller/flp/flp_test.go | 4 +- internal/controller/flp/metrics_api_test.go | 2 +- internal/pkg/helper/loki_config.go | 49 +- .../flowlogs-pipeline/pkg/api/encode_prom.go | 1 - .../flowlogs-pipeline/pkg/api/write_loki.go | 73 +- vendor/modules.txt | 5 + 22 files changed, 1198 insertions(+), 64 deletions(-) create mode 100644 catalog/unreleased-legacy/v1.9.2-community/bundle.yaml create mode 100644 catalog/unreleased-legacy/v1.9.2-community/index.yaml create mode 100644 catalog/unreleased-legacy/v1.9.2-community/other.yaml create mode 100644 catalog/unreleased/v1.9.2-community/bundle.yaml create mode 100644 catalog/unreleased/v1.9.2-community/index.yaml create mode 100644 catalog/unreleased/v1.9.2-community/other.yaml diff --git a/api/flowcollector/v1beta2/helper.go b/api/flowcollector/v1beta2/helper.go index bbd5a93b5..92f3945e6 100644 --- a/api/flowcollector/v1beta2/helper.go +++ b/api/flowcollector/v1beta2/helper.go @@ -196,3 +196,12 @@ func (spec *FlowCollectorSpec) HasExperimentalAlertsHealth() bool { func (spec *FlowCollectorSpec) DeployNetworkPolicy() bool { return spec.NetworkPolicy.Enable != nil && *spec.NetworkPolicy.Enable } + +func (spec *FlowCollectorFLP) HasExperimentalLokiGRPCClientProtocol() bool { + if spec.Advanced != nil { + env := spec.Advanced.Env["LOKI_USE_GRPC_CLIENT_PROTOCOL"] + useGRPC, err := strconv.ParseBool(env) + return err == nil && useGRPC + } + return false +} diff --git a/bundle/manifests/netobserv-operator.clusterserviceversion.yaml b/bundle/manifests/netobserv-operator.clusterserviceversion.yaml index 6a3c914de..58f703cfa 100644 --- a/bundle/manifests/netobserv-operator.clusterserviceversion.yaml +++ b/bundle/manifests/netobserv-operator.clusterserviceversion.yaml @@ -480,6 +480,8 @@ spec: x-descriptors: - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Read timeout + path: loki.readTimeout - path: loki.advanced x-descriptors: - urn:alm:descriptor:com.tectonic.ui:hidden @@ -596,8 +598,6 @@ spec: path: loki.monolithic.tenantID - displayName: Url path: loki.monolithic.url - - displayName: Read timeout - path: loki.readTimeout - displayName: Namespace path: namespace - displayName: Network policy @@ -658,7 +658,7 @@ spec: - urn:alm:descriptor:io.kubernetes.conditions version: v1beta2 - description: '`FlowMetric` is the schema for the custom metrics API, which allows - to generate more metrics out of flow logs. You can find examples here: https://github.com/netobserv/network-observability-operator/tree/main/config/samples/flowmetrics' + to generate more metrics out of flow logs. You can find examples here: https://github.com/netobserv/network-observability-operator/tree/v1.9.2-community/config/samples/flowmetrics' displayName: Flow Metric kind: FlowMetric name: flowmetrics.flows.netobserv.io diff --git a/catalog/unreleased-legacy/v1.9.2-community/bundle.yaml b/catalog/unreleased-legacy/v1.9.2-community/bundle.yaml new file mode 100644 index 000000000..4263ff2a0 --- /dev/null +++ b/catalog/unreleased-legacy/v1.9.2-community/bundle.yaml @@ -0,0 +1,84 @@ +--- +image: quay.io/netobserv/network-observability-operator-bundle:v1.9.2-community +name: netobserv-operator.v1.9.2-community +package: netobserv-operator +properties: +- type: olm.gvk + value: + group: flows.netobserv.io + kind: FlowCollector + version: v1beta2 +- type: olm.gvk + value: + group: flows.netobserv.io + kind: FlowMetric + version: v1alpha1 +- type: olm.package + value: + packageName: netobserv-operator + version: 1.9.2-community +- type: olm.bundle.object + value: + data: {"apiVersion":"apiextensions.k8s.io/v1","kind":"CustomResourceDefinition","metadata":{"annotations":{"controller-gen.kubebuilder.io/version":"v0.16.2"},"creationTimestamp":null,"name":"flowcollectors.flows.netobserv.io"},"spec":{"group":"flows.netobserv.io","names":{"kind":"FlowCollector","listKind":"FlowCollectorList","plural":"flowcollectors","singular":"flowcollector"},"scope":"Cluster","versions":[{"additionalPrinterColumns":[{"jsonPath":".spec.agent.type","name":"Agent","type":"string"},{"jsonPath":".spec.agent.ebpf.sampling","name":"Sampling (EBPF)","type":"string"},{"jsonPath":".spec.deploymentModel","name":"Deployment Model","type":"string"},{"jsonPath":".status.conditions[?(@.type==\"Ready\")].reason","name":"Status","type":"string"},{"jsonPath":".status.conditions[?(@.type==\"ConfigurationIssue\")].message","name":"Warnings","type":"string"}],"name":"v1beta2","schema":{"openAPIV3Schema":{"description":"`FlowCollector` is the schema for the network flows collection API, which pilots and configures the underlying deployments.","properties":{"apiVersion":{"description":"APIVersion defines the versioned schema of this representation of an object.\nServers should convert recognized schemas to the latest internal value, and\nmay reject unrecognized values.\nMore info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources","type":"string"},"kind":{"description":"Kind is a string value representing the REST resource this object represents.\nServers may infer this from the endpoint the client submits requests to.\nCannot be updated.\nIn CamelCase.\nMore info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds","type":"string"},"metadata":{"properties":{"name":{"pattern":"^cluster$","type":"string"}},"type":"object"},"spec":{"description":"Defines the desired state of the FlowCollector resource.\n\u003cbr\u003e\u003cbr\u003e\n*: the mention of \"unsupported\" or \"deprecated\" for a feature throughout this document means that this feature\nis not officially supported by Red Hat. It might have been, for example, contributed by the community\nand accepted without a formal agreement for maintenance. The product maintainers might provide some support\nfor these features as a best effort only.","properties":{"agent":{"description":"Agent configuration for flows extraction.","properties":{"ebpf":{"description":"`ebpf` describes the settings related to the eBPF-based flow reporter when `spec.agent.type`\nis set to `eBPF`.","properties":{"advanced":{"description":"`advanced` allows setting some aspects of the internal configuration of the eBPF agent.\nThis section is aimed mostly for debugging and fine-grained performance optimizations,\nsuch as `GOGC` and `GOMAXPROCS` environment variables. Set these values at your own risk. You can also\noverride the default Linux capabilities from there.","properties":{"capOverride":{"description":"Linux capabilities override, when not running as privileged. Default capabilities are BPF, PERFMON and NET_ADMIN.","items":{"type":"string"},"type":"array"},"env":{"additionalProperties":{"type":"string"},"description":"`env` allows passing custom environment variables to underlying components. Useful for passing\nsome very concrete performance-tuning options, such as `GOGC` and `GOMAXPROCS`, that should not be\npublicly exposed as part of the FlowCollector descriptor, as they are only useful\nin edge debug or support scenarios.","type":"object"},"scheduling":{"description":"scheduling controls how the pods are scheduled on nodes.","properties":{"affinity":{"description":"If specified, the pod's scheduling constraints. For documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#scheduling.","properties":{"nodeAffinity":{"description":"Describes node affinity scheduling rules for the pod.","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and adding\n\"weight\" to the sum if the node matches the corresponding matchExpressions; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"An empty preferred scheduling term matches all objects with implicit weight 0\n(i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op).","properties":{"preference":{"description":"A node selector term, associated with the corresponding weight.","properties":{"matchExpressions":{"description":"A list of node selector requirements by node's labels.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchFields":{"description":"A list of node selector requirements by node's fields.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object","x-kubernetes-map-type":"atomic"},"weight":{"description":"Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100.","format":"int32","type":"integer"}},"required":["preference","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to an update), the system\nmay or may not try to eventually evict the pod from its node.","properties":{"nodeSelectorTerms":{"description":"Required. A list of node selector terms. The terms are ORed.","items":{"description":"A null or empty node selector term matches no objects. The requirements of\nthem are ANDed.\nThe TopologySelectorTerm type implements a subset of the NodeSelectorTerm.","properties":{"matchExpressions":{"description":"A list of node selector requirements by node's labels.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchFields":{"description":"A list of node selector requirements by node's fields.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object","x-kubernetes-map-type":"atomic"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["nodeSelectorTerms"],"type":"object","x-kubernetes-map-type":"atomic"}},"type":"object"},"podAffinity":{"description":"Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)).","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and adding\n\"weight\" to the sum if the node has pods which matches the corresponding podAffinityTerm; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s)","properties":{"podAffinityTerm":{"description":"Required. A pod affinity term, associated with the corresponding weight.","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"weight":{"description":"weight associated with matching the corresponding podAffinityTerm,\nin the range 1-100.","format":"int32","type":"integer"}},"required":["podAffinityTerm","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to a pod label update), the\nsystem may or may not try to eventually evict the pod from its node.\nWhen there are multiple elements, the lists of nodes corresponding to each\npodAffinityTerm are intersected, i.e. all terms must be satisfied.","items":{"description":"Defines a set of pods (namely those matching the labelSelector\nrelative to the given namespace(s)) that this pod should be\nco-located (affinity) or not co-located (anti-affinity) with,\nwhere co-located is defined as running on a node whose value of\nthe label with key \u003ctopologyKey\u003e matches that of any node on which\na pod of the set of pods is running","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object"},"podAntiAffinity":{"description":"Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)).","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe anti-affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling anti-affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and subtracting\n\"weight\" from the sum if the node has pods which matches the corresponding podAffinityTerm; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s)","properties":{"podAffinityTerm":{"description":"Required. A pod affinity term, associated with the corresponding weight.","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"weight":{"description":"weight associated with matching the corresponding podAffinityTerm,\nin the range 1-100.","format":"int32","type":"integer"}},"required":["podAffinityTerm","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the anti-affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the anti-affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to a pod label update), the\nsystem may or may not try to eventually evict the pod from its node.\nWhen there are multiple elements, the lists of nodes corresponding to each\npodAffinityTerm are intersected, i.e. all terms must be satisfied.","items":{"description":"Defines a set of pods (namely those matching the labelSelector\nrelative to the given namespace(s)) that this pod should be\nco-located (affinity) or not co-located (anti-affinity) with,\nwhere co-located is defined as running on a node whose value of\nthe label with key \u003ctopologyKey\u003e matches that of any node on which\na pod of the set of pods is running","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object"}},"type":"object"},"nodeSelector":{"additionalProperties":{"type":"string"},"description":"`nodeSelector` allows scheduling of pods only onto nodes that have each of the specified labels.\nFor documentation, refer to https://kubernetes.io/docs/concepts/configuration/assign-pod-node/.","type":"object","x-kubernetes-map-type":"atomic"},"priorityClassName":{"description":"If specified, indicates the pod's priority. For documentation, refer to https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/#how-to-use-priority-and-preemption.\nIf not specified, default priority is used, or zero if there is no default.","type":"string"},"tolerations":{"description":"`tolerations` is a list of tolerations that allow the pod to schedule onto nodes with matching taints.\nFor documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#scheduling.","items":{"description":"The pod this Toleration is attached to tolerates any taint that matches\nthe triple \u003ckey,value,effect\u003e using the matching operator \u003coperator\u003e.","properties":{"effect":{"description":"Effect indicates the taint effect to match. Empty means match all taint effects.\nWhen specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute.","type":"string"},"key":{"description":"Key is the taint key that the toleration applies to. Empty means match all taint keys.\nIf the key is empty, operator must be Exists; this combination means to match all values and all keys.","type":"string"},"operator":{"description":"Operator represents a key's relationship to the value.\nValid operators are Exists and Equal. Defaults to Equal.\nExists is equivalent to wildcard for value, so that a pod can\ntolerate all taints of a particular category.","type":"string"},"tolerationSeconds":{"description":"TolerationSeconds represents the period of time the toleration (which must be\nof effect NoExecute, otherwise this field is ignored) tolerates the taint. By default,\nit is not set, which means tolerate the taint forever (do not evict). Zero and\nnegative values will be treated as 0 (evict immediately) by the system.","format":"int64","type":"integer"},"value":{"description":"Value is the taint value the toleration matches to.\nIf the operator is Exists, the value should be empty, otherwise just a regular string.","type":"string"}},"type":"object"},"type":"array"}},"type":"object"}},"type":"object"},"cacheActiveTimeout":{"default":"5s","description":"`cacheActiveTimeout` is the max period during which the reporter aggregates flows before sending.\nIncreasing `cacheMaxFlows` and `cacheActiveTimeout` can decrease the network traffic overhead and the CPU load,\nhowever you can expect higher memory consumption and an increased latency in the flow collection.","pattern":"^\\d+(ns|ms|s|m)?$","type":"string"},"cacheMaxFlows":{"default":100000,"description":"`cacheMaxFlows` is the max number of flows in an aggregate; when reached, the reporter sends the flows.\nIncreasing `cacheMaxFlows` and `cacheActiveTimeout` can decrease the network traffic overhead and the CPU load,\nhowever you can expect higher memory consumption and an increased latency in the flow collection.","format":"int32","minimum":1,"type":"integer"},"excludeInterfaces":{"default":["lo"],"description":"`excludeInterfaces` contains the interface names that are excluded from flow tracing.\nAn entry enclosed by slashes, such as `/br-/`, is matched as a regular expression.\nOtherwise it is matched as a case-sensitive string.","items":{"type":"string"},"type":"array"},"features":{"description":"List of additional features to enable. They are all disabled by default. Enabling additional features might have performance impacts. Possible values are:\u003cbr\u003e\n- `PacketDrop`: Enable the packets drop flows logging feature. This feature requires mounting\nthe kernel debug filesystem, so the eBPF agent pods must run as privileged via `spec.agent.ebpf.privileged`.\u003cbr\u003e\n- `DNSTracking`: Enable the DNS tracking feature.\u003cbr\u003e\n- `FlowRTT`: Enable flow latency (sRTT) extraction in the eBPF agent from TCP traffic.\u003cbr\u003e\n- `NetworkEvents`: Enable the network events monitoring feature, such as correlating flows and network policies.\nThis feature requires mounting the kernel debug filesystem, so the eBPF agent pods must run as privileged via `spec.agent.ebpf.privileged`.\nIt requires using the OVN-Kubernetes network plugin with the Observability feature.\nIMPORTANT: This feature is available as a Technology Preview.\u003cbr\u003e\n- `PacketTranslation`: Enable enriching flows with packet translation information, such as Service NAT.\u003cbr\u003e\n- `EbpfManager`: [Unsupported (*)]. Use eBPF Manager to manage NetObserv eBPF programs. Pre-requisite: the eBPF Manager operator (or upstream bpfman operator) must be installed.\u003cbr\u003e\n- `UDNMapping`: Enable interfaces mapping to User Defined Networks (UDN). \u003cbr\u003e\nThis feature requires mounting the kernel debug filesystem, so the eBPF agent pods must run as privileged via `spec.agent.ebpf.privileged`.\nIt requires using the OVN-Kubernetes network plugin with the Observability feature. \u003cbr\u003e\n- `IPSec`, to track flows between nodes with IPsec encryption. \u003cbr\u003e","items":{"description":"Agent feature, can be one of:\u003cbr\u003e\n- `PacketDrop`, to track packet drops.\u003cbr\u003e\n- `DNSTracking`, to track specific information on DNS traffic.\u003cbr\u003e\n- `FlowRTT`, to track TCP latency.\u003cbr\u003e\n- `NetworkEvents`, to track network events [Technology Preview].\u003cbr\u003e\n- `PacketTranslation`, to enrich flows with packets translation information, such as Service NAT.\u003cbr\u003e\n- `EbpfManager`, to enable using eBPF Manager to manage NetObserv eBPF programs. [Unsupported (*)].\u003cbr\u003e\n- `UDNMapping`, to enable interfaces mapping to UDN. \u003cbr\u003e\n- `IPSec`, to track flows between nodes with IPsec encryption. \u003cbr\u003e","enum":["PacketDrop","DNSTracking","FlowRTT","NetworkEvents","PacketTranslation","EbpfManager","UDNMapping","IPSec"],"type":"string"},"type":"array"},"flowFilter":{"description":"`flowFilter` defines the eBPF agent configuration regarding flow filtering.","properties":{"action":{"description":"`action` defines the action to perform on the flows that match the filter. The available options are `Accept`, which is the default, and `Reject`.","enum":["Accept","Reject"],"type":"string"},"cidr":{"description":"`cidr` defines the IP CIDR to filter flows by.\nExamples: `10.10.10.0/24` or `100:100:100:100::/64`","type":"string"},"destPorts":{"anyOf":[{"type":"integer"},{"type":"string"}],"description":"`destPorts` optionally defines the destination ports to filter flows by.\nTo filter a single port, set a single port as an integer value. For example, `destPorts: 80`.\nTo filter a range of ports, use a \"start-end\" range in string format. For example, `destPorts: \"80-100\"`.\nTo filter two ports, use a \"port1,port2\" in string format. For example, `ports: \"80,100\"`.","x-kubernetes-int-or-string":true},"direction":{"description":"`direction` optionally defines a direction to filter flows by. The available options are `Ingress` and `Egress`.","enum":["Ingress","Egress"],"type":"string"},"enable":{"description":"Set `enable` to `true` to enable the eBPF flow filtering feature.","type":"boolean"},"icmpCode":{"description":"`icmpCode`, for Internet Control Message Protocol (ICMP) traffic, optionally defines the ICMP code to filter flows by.","type":"integer"},"icmpType":{"description":"`icmpType`, for ICMP traffic, optionally defines the ICMP type to filter flows by.","type":"integer"},"peerCIDR":{"description":"`peerCIDR` defines the Peer IP CIDR to filter flows by.\nExamples: `10.10.10.0/24` or `100:100:100:100::/64`","type":"string"},"peerIP":{"description":"`peerIP` optionally defines the remote IP address to filter flows by.\nExample: `10.10.10.10`.","type":"string"},"pktDrops":{"description":"`pktDrops` optionally filters only flows containing packet drops.","type":"boolean"},"ports":{"anyOf":[{"type":"integer"},{"type":"string"}],"description":"`ports` optionally defines the ports to filter flows by. It is used both for source and destination ports.\nTo filter a single port, set a single port as an integer value. For example, `ports: 80`.\nTo filter a range of ports, use a \"start-end\" range in string format. For example, `ports: \"80-100\"`.\nTo filter two ports, use a \"port1,port2\" in string format. For example, `ports: \"80,100\"`.","x-kubernetes-int-or-string":true},"protocol":{"description":"`protocol` optionally defines a protocol to filter flows by. The available options are `TCP`, `UDP`, `ICMP`, `ICMPv6`, and `SCTP`.","enum":["TCP","UDP","ICMP","ICMPv6","SCTP"],"type":"string"},"rules":{"description":"`rules` defines a list of filtering rules on the eBPF Agents.\nWhen filtering is enabled, by default, flows that don't match any rule are rejected.\nTo change the default, you can define a rule that accepts everything: `{ action: \"Accept\", cidr: \"0.0.0.0/0\" }`, and then refine with rejecting rules.","items":{"description":"`EBPFFlowFilterRule` defines the desired eBPF agent configuration regarding flow filtering rule.","properties":{"action":{"description":"`action` defines the action to perform on the flows that match the filter. The available options are `Accept`, which is the default, and `Reject`.","enum":["Accept","Reject"],"type":"string"},"cidr":{"description":"`cidr` defines the IP CIDR to filter flows by.\nExamples: `10.10.10.0/24` or `100:100:100:100::/64`","type":"string"},"destPorts":{"anyOf":[{"type":"integer"},{"type":"string"}],"description":"`destPorts` optionally defines the destination ports to filter flows by.\nTo filter a single port, set a single port as an integer value. For example, `destPorts: 80`.\nTo filter a range of ports, use a \"start-end\" range in string format. For example, `destPorts: \"80-100\"`.\nTo filter two ports, use a \"port1,port2\" in string format. For example, `ports: \"80,100\"`.","x-kubernetes-int-or-string":true},"direction":{"description":"`direction` optionally defines a direction to filter flows by. The available options are `Ingress` and `Egress`.","enum":["Ingress","Egress"],"type":"string"},"icmpCode":{"description":"`icmpCode`, for Internet Control Message Protocol (ICMP) traffic, optionally defines the ICMP code to filter flows by.","type":"integer"},"icmpType":{"description":"`icmpType`, for ICMP traffic, optionally defines the ICMP type to filter flows by.","type":"integer"},"peerCIDR":{"description":"`peerCIDR` defines the Peer IP CIDR to filter flows by.\nExamples: `10.10.10.0/24` or `100:100:100:100::/64`","type":"string"},"peerIP":{"description":"`peerIP` optionally defines the remote IP address to filter flows by.\nExample: `10.10.10.10`.","type":"string"},"pktDrops":{"description":"`pktDrops` optionally filters only flows containing packet drops.","type":"boolean"},"ports":{"anyOf":[{"type":"integer"},{"type":"string"}],"description":"`ports` optionally defines the ports to filter flows by. It is used both for source and destination ports.\nTo filter a single port, set a single port as an integer value. For example, `ports: 80`.\nTo filter a range of ports, use a \"start-end\" range in string format. For example, `ports: \"80-100\"`.\nTo filter two ports, use a \"port1,port2\" in string format. For example, `ports: \"80,100\"`.","x-kubernetes-int-or-string":true},"protocol":{"description":"`protocol` optionally defines a protocol to filter flows by. The available options are `TCP`, `UDP`, `ICMP`, `ICMPv6`, and `SCTP`.","enum":["TCP","UDP","ICMP","ICMPv6","SCTP"],"type":"string"},"sampling":{"description":"`sampling` is the sampling ratio for the matched packets, overriding the global sampling defined at `spec.agent.ebpf.sampling`.","format":"int32","type":"integer"},"sourcePorts":{"anyOf":[{"type":"integer"},{"type":"string"}],"description":"`sourcePorts` optionally defines the source ports to filter flows by.\nTo filter a single port, set a single port as an integer value. For example, `sourcePorts: 80`.\nTo filter a range of ports, use a \"start-end\" range in string format. For example, `sourcePorts: \"80-100\"`.\nTo filter two ports, use a \"port1,port2\" in string format. For example, `ports: \"80,100\"`.","x-kubernetes-int-or-string":true},"tcpFlags":{"description":"`tcpFlags` optionally defines TCP flags to filter flows by.\nIn addition to the standard flags (RFC-9293), you can also filter by one of the three following combinations: `SYN-ACK`, `FIN-ACK`, and `RST-ACK`.","enum":["SYN","SYN-ACK","ACK","FIN","RST","URG","ECE","CWR","FIN-ACK","RST-ACK"],"type":"string"}},"type":"object"},"maxItems":16,"minItems":1,"type":"array"},"sampling":{"description":"`sampling` is the sampling ratio for the matched packets, overriding the global sampling defined at `spec.agent.ebpf.sampling`.","format":"int32","type":"integer"},"sourcePorts":{"anyOf":[{"type":"integer"},{"type":"string"}],"description":"`sourcePorts` optionally defines the source ports to filter flows by.\nTo filter a single port, set a single port as an integer value. For example, `sourcePorts: 80`.\nTo filter a range of ports, use a \"start-end\" range in string format. For example, `sourcePorts: \"80-100\"`.\nTo filter two ports, use a \"port1,port2\" in string format. For example, `ports: \"80,100\"`.","x-kubernetes-int-or-string":true},"tcpFlags":{"description":"`tcpFlags` optionally defines TCP flags to filter flows by.\nIn addition to the standard flags (RFC-9293), you can also filter by one of the three following combinations: `SYN-ACK`, `FIN-ACK`, and `RST-ACK`.","enum":["SYN","SYN-ACK","ACK","FIN","RST","URG","ECE","CWR","FIN-ACK","RST-ACK"],"type":"string"}},"type":"object"},"imagePullPolicy":{"default":"IfNotPresent","description":"`imagePullPolicy` is the Kubernetes pull policy for the image defined above","enum":["IfNotPresent","Always","Never"],"type":"string"},"interfaces":{"description":"`interfaces` contains the interface names from where flows are collected. If empty, the agent\nfetches all the interfaces in the system, excepting the ones listed in `excludeInterfaces`.\nAn entry enclosed by slashes, such as `/br-/`, is matched as a regular expression.\nOtherwise it is matched as a case-sensitive string.","items":{"type":"string"},"type":"array"},"kafkaBatchSize":{"default":1048576,"description":"`kafkaBatchSize` limits the maximum size of a request in bytes before being sent to a partition. Ignored when not using Kafka. Default: 1MB.","type":"integer"},"logLevel":{"default":"info","description":"`logLevel` defines the log level for the NetObserv eBPF Agent","enum":["trace","debug","info","warn","error","fatal","panic"],"type":"string"},"metrics":{"description":"`metrics` defines the eBPF agent configuration regarding metrics.","properties":{"disableAlerts":{"description":"`disableAlerts` is a list of alerts that should be disabled.\nPossible values are:\u003cbr\u003e\n`NetObservDroppedFlows`, which is triggered when the eBPF agent is missing packets or flows, such as when the BPF hashmap is busy or full, or the capacity limiter is being triggered.\u003cbr\u003e","items":{"description":"Name of an eBPF agent alert.\nPossible values are:\u003cbr\u003e\n`NetObservDroppedFlows`, which is triggered when the eBPF agent is missing packets or flows, such as when the BPF hashmap is busy or full, or the capacity limiter is being triggered.\u003cbr\u003e","enum":["NetObservDroppedFlows"],"type":"string"},"type":"array"},"enable":{"description":"Set `enable` to `false` to disable eBPF agent metrics collection. It is enabled by default.","type":"boolean"},"server":{"description":"Metrics server endpoint configuration for the Prometheus scraper.","properties":{"port":{"description":"The metrics server HTTP port.","format":"int32","maximum":65535,"minimum":1,"type":"integer"},"tls":{"description":"TLS configuration.","properties":{"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the provided certificate.\nIf set to `true`, the `providedCaFile` field is ignored.","type":"boolean"},"provided":{"description":"TLS configuration when `type` is set to `Provided`.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"providedCaFile":{"description":"Reference to the CA file when `type` is set to `Provided`.","properties":{"file":{"description":"File name within the config map or secret.","type":"string"},"name":{"description":"Name of the config map or secret containing the file.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing the file. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the file reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"type":{"default":"Disabled","description":"Select the type of TLS configuration:\u003cbr\u003e\n- `Disabled` (default) to not configure TLS for the endpoint.\n- `Provided` to manually provide cert file and a key file. [Unsupported (*)].\n- `Auto` to use OpenShift auto generated certificate using annotations.","enum":["Disabled","Provided","Auto"],"type":"string"}},"required":["type"],"type":"object"}},"type":"object"}},"type":"object"},"privileged":{"description":"Privileged mode for the eBPF Agent container. When ignored or set to `false`, the operator sets\ngranular capabilities (BPF, PERFMON, NET_ADMIN) to the container.\nIf for some reason these capabilities cannot be set, such as if an old kernel version not knowing CAP_BPF\nis in use, then you can turn on this mode for more global privileges.\nSome agent features require the privileged mode, such as packet drops tracking (see `features`) and SR-IOV support.","type":"boolean"},"resources":{"default":{"limits":{"memory":"800Mi"},"requests":{"cpu":"100m","memory":"50Mi"}},"description":"`resources` are the compute resources required by this container.\nFor more information, see https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","properties":{"claims":{"description":"Claims lists the names of resources, defined in spec.resourceClaims,\nthat are used by this container.\n\nThis field depends on the\nDynamicResourceAllocation feature gate.\n\nThis field is immutable. It can only be set for containers.","items":{"description":"ResourceClaim references one entry in PodSpec.ResourceClaims.","properties":{"name":{"description":"Name must match the name of one entry in pod.spec.resourceClaims of\nthe Pod where this field is used. It makes that resource available\ninside a container.","type":"string"},"request":{"description":"Request is the name chosen for a request in the referenced claim.\nIf empty, everything from the claim is made available, otherwise\nonly the result of this request.","type":"string"}},"required":["name"],"type":"object"},"type":"array","x-kubernetes-list-map-keys":["name"],"x-kubernetes-list-type":"map"},"limits":{"additionalProperties":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"description":"Limits describes the maximum amount of compute resources allowed.\nMore info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","type":"object"},"requests":{"additionalProperties":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"description":"Requests describes the minimum amount of compute resources required.\nIf Requests is omitted for a container, it defaults to Limits if that is explicitly specified,\notherwise to an implementation-defined value. Requests cannot exceed Limits.\nMore info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","type":"object"}},"type":"object"},"sampling":{"default":50,"description":"Sampling ratio of the eBPF probe. 100 means one packet on 100 is sent. 0 or 1 means all packets are sampled.","format":"int32","minimum":0,"type":"integer"}},"type":"object"},"ipfix":{"description":"`ipfix` [deprecated (*)] - describes the settings related to the IPFIX-based flow reporter when `spec.agent.type`\nis set to `IPFIX`.","properties":{"cacheActiveTimeout":{"default":"20s","description":"`cacheActiveTimeout` is the max period during which the reporter aggregates flows before sending.","pattern":"^\\d+(ns|ms|s|m)?$","type":"string"},"cacheMaxFlows":{"default":400,"description":"`cacheMaxFlows` is the max number of flows in an aggregate; when reached, the reporter sends the flows.","format":"int32","minimum":0,"type":"integer"},"clusterNetworkOperator":{"description":"`clusterNetworkOperator` defines the settings related to the OpenShift Cluster Network Operator, when available.","properties":{"namespace":{"default":"openshift-network-operator","description":"Namespace  where the config map is going to be deployed.","type":"string"}},"type":"object"},"forceSampleAll":{"default":false,"description":"`forceSampleAll` allows disabling sampling in the IPFIX-based flow reporter.\nIt is not recommended to sample all the traffic with IPFIX, as it might generate cluster instability.\nIf you REALLY want to do that, set this flag to `true`. Use at your own risk.\nWhen it is set to `true`, the value of `sampling` is ignored.","type":"boolean"},"ovnKubernetes":{"description":"`ovnKubernetes` defines the settings of the OVN-Kubernetes network plugin, when available. This configuration is used when using OVN's IPFIX exports, without OpenShift. When using OpenShift, refer to the `clusterNetworkOperator` property instead.","properties":{"containerName":{"default":"ovnkube-node","description":"`containerName` defines the name of the container to configure for IPFIX.","type":"string"},"daemonSetName":{"default":"ovnkube-node","description":"`daemonSetName` defines the name of the DaemonSet controlling the OVN-Kubernetes pods.","type":"string"},"namespace":{"default":"ovn-kubernetes","description":"Namespace where OVN-Kubernetes pods are deployed.","type":"string"}},"type":"object"},"sampling":{"default":400,"description":"`sampling` is the sampling rate on the reporter. 100 means one flow on 100 is sent.\nTo ensure cluster stability, it is not possible to set a value below 2.\nIf you really want to sample every packet, which might impact the cluster stability,\nrefer to `forceSampleAll`. Alternatively, you can use the eBPF Agent instead of IPFIX.","format":"int32","minimum":2,"type":"integer"}},"type":"object"},"type":{"default":"eBPF","description":"`type` [deprecated (*)] selects the flows tracing agent. Previously, this field allowed to select between `eBPF` or `IPFIX`.\nOnly `eBPF` is allowed now, so this field is deprecated and is planned for removal in a future version of the API.","enum":["eBPF","IPFIX"],"type":"string"}},"type":"object"},"consolePlugin":{"description":"`consolePlugin` defines the settings related to the OpenShift Console plugin, when available.","properties":{"advanced":{"description":"`advanced` allows setting some aspects of the internal configuration of the console plugin.\nThis section is aimed mostly for debugging and fine-grained performance optimizations,\nsuch as `GOGC` and `GOMAXPROCS` environment variables. Set these values at your own risk.","properties":{"args":{"description":"`args` allows passing custom arguments to underlying components. Useful for overriding\nsome parameters, such as a URL or a configuration path, that should not be\npublicly exposed as part of the FlowCollector descriptor, as they are only useful\nin edge debug or support scenarios.","items":{"type":"string"},"type":"array"},"env":{"additionalProperties":{"type":"string"},"description":"`env` allows passing custom environment variables to underlying components. Useful for passing\nsome very concrete performance-tuning options, such as `GOGC` and `GOMAXPROCS`, that should not be\npublicly exposed as part of the FlowCollector descriptor, as they are only useful\nin edge debug or support scenarios.","type":"object"},"port":{"default":9001,"description":"`port` is the plugin service port. Do not use 9002, which is reserved for metrics.","format":"int32","maximum":65535,"minimum":1,"type":"integer"},"register":{"default":true,"description":"`register` allows, when set to `true`, to automatically register the provided console plugin with the OpenShift Console operator.\nWhen set to `false`, you can still register it manually by editing console.operator.openshift.io/cluster with the following command:\n`oc patch console.operator.openshift.io cluster --type='json' -p '[{\"op\": \"add\", \"path\": \"/spec/plugins/-\", \"value\": \"netobserv-plugin\"}]'`","type":"boolean"},"scheduling":{"description":"`scheduling` controls how the pods are scheduled on nodes.","properties":{"affinity":{"description":"If specified, the pod's scheduling constraints. For documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#scheduling.","properties":{"nodeAffinity":{"description":"Describes node affinity scheduling rules for the pod.","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and adding\n\"weight\" to the sum if the node matches the corresponding matchExpressions; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"An empty preferred scheduling term matches all objects with implicit weight 0\n(i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op).","properties":{"preference":{"description":"A node selector term, associated with the corresponding weight.","properties":{"matchExpressions":{"description":"A list of node selector requirements by node's labels.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchFields":{"description":"A list of node selector requirements by node's fields.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object","x-kubernetes-map-type":"atomic"},"weight":{"description":"Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100.","format":"int32","type":"integer"}},"required":["preference","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to an update), the system\nmay or may not try to eventually evict the pod from its node.","properties":{"nodeSelectorTerms":{"description":"Required. A list of node selector terms. The terms are ORed.","items":{"description":"A null or empty node selector term matches no objects. The requirements of\nthem are ANDed.\nThe TopologySelectorTerm type implements a subset of the NodeSelectorTerm.","properties":{"matchExpressions":{"description":"A list of node selector requirements by node's labels.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchFields":{"description":"A list of node selector requirements by node's fields.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object","x-kubernetes-map-type":"atomic"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["nodeSelectorTerms"],"type":"object","x-kubernetes-map-type":"atomic"}},"type":"object"},"podAffinity":{"description":"Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)).","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and adding\n\"weight\" to the sum if the node has pods which matches the corresponding podAffinityTerm; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s)","properties":{"podAffinityTerm":{"description":"Required. A pod affinity term, associated with the corresponding weight.","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"weight":{"description":"weight associated with matching the corresponding podAffinityTerm,\nin the range 1-100.","format":"int32","type":"integer"}},"required":["podAffinityTerm","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to a pod label update), the\nsystem may or may not try to eventually evict the pod from its node.\nWhen there are multiple elements, the lists of nodes corresponding to each\npodAffinityTerm are intersected, i.e. all terms must be satisfied.","items":{"description":"Defines a set of pods (namely those matching the labelSelector\nrelative to the given namespace(s)) that this pod should be\nco-located (affinity) or not co-located (anti-affinity) with,\nwhere co-located is defined as running on a node whose value of\nthe label with key \u003ctopologyKey\u003e matches that of any node on which\na pod of the set of pods is running","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object"},"podAntiAffinity":{"description":"Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)).","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe anti-affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling anti-affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and subtracting\n\"weight\" from the sum if the node has pods which matches the corresponding podAffinityTerm; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s)","properties":{"podAffinityTerm":{"description":"Required. A pod affinity term, associated with the corresponding weight.","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"weight":{"description":"weight associated with matching the corresponding podAffinityTerm,\nin the range 1-100.","format":"int32","type":"integer"}},"required":["podAffinityTerm","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the anti-affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the anti-affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to a pod label update), the\nsystem may or may not try to eventually evict the pod from its node.\nWhen there are multiple elements, the lists of nodes corresponding to each\npodAffinityTerm are intersected, i.e. all terms must be satisfied.","items":{"description":"Defines a set of pods (namely those matching the labelSelector\nrelative to the given namespace(s)) that this pod should be\nco-located (affinity) or not co-located (anti-affinity) with,\nwhere co-located is defined as running on a node whose value of\nthe label with key \u003ctopologyKey\u003e matches that of any node on which\na pod of the set of pods is running","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object"}},"type":"object"},"nodeSelector":{"additionalProperties":{"type":"string"},"description":"`nodeSelector` allows scheduling of pods only onto nodes that have each of the specified labels.\nFor documentation, refer to https://kubernetes.io/docs/concepts/configuration/assign-pod-node/.","type":"object","x-kubernetes-map-type":"atomic"},"priorityClassName":{"description":"If specified, indicates the pod's priority. For documentation, refer to https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/#how-to-use-priority-and-preemption.\nIf not specified, default priority is used, or zero if there is no default.","type":"string"},"tolerations":{"description":"`tolerations` is a list of tolerations that allow the pod to schedule onto nodes with matching taints.\nFor documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#scheduling.","items":{"description":"The pod this Toleration is attached to tolerates any taint that matches\nthe triple \u003ckey,value,effect\u003e using the matching operator \u003coperator\u003e.","properties":{"effect":{"description":"Effect indicates the taint effect to match. Empty means match all taint effects.\nWhen specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute.","type":"string"},"key":{"description":"Key is the taint key that the toleration applies to. Empty means match all taint keys.\nIf the key is empty, operator must be Exists; this combination means to match all values and all keys.","type":"string"},"operator":{"description":"Operator represents a key's relationship to the value.\nValid operators are Exists and Equal. Defaults to Equal.\nExists is equivalent to wildcard for value, so that a pod can\ntolerate all taints of a particular category.","type":"string"},"tolerationSeconds":{"description":"TolerationSeconds represents the period of time the toleration (which must be\nof effect NoExecute, otherwise this field is ignored) tolerates the taint. By default,\nit is not set, which means tolerate the taint forever (do not evict). Zero and\nnegative values will be treated as 0 (evict immediately) by the system.","format":"int64","type":"integer"},"value":{"description":"Value is the taint value the toleration matches to.\nIf the operator is Exists, the value should be empty, otherwise just a regular string.","type":"string"}},"type":"object"},"type":"array"}},"type":"object"}},"type":"object"},"autoscaler":{"description":"`autoscaler` spec of a horizontal pod autoscaler to set up for the plugin Deployment.","properties":{"maxReplicas":{"default":3,"description":"`maxReplicas` is the upper limit for the number of pods that can be set by the autoscaler; cannot be smaller than MinReplicas.","format":"int32","type":"integer"},"metrics":{"description":"Metrics used by the pod autoscaler. For documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/horizontal-pod-autoscaler-v2/","items":{"properties":{"containerResource":{"properties":{"container":{"type":"string"},"name":{"type":"string"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["container","name","target"],"type":"object"},"external":{"properties":{"metric":{"properties":{"name":{"type":"string"},"selector":{"properties":{"matchExpressions":{"items":{"properties":{"key":{"type":"string"},"operator":{"type":"string"},"values":{"items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"}},"required":["name"],"type":"object"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["metric","target"],"type":"object"},"object":{"properties":{"describedObject":{"properties":{"apiVersion":{"type":"string"},"kind":{"type":"string"},"name":{"type":"string"}},"required":["kind","name"],"type":"object"},"metric":{"properties":{"name":{"type":"string"},"selector":{"properties":{"matchExpressions":{"items":{"properties":{"key":{"type":"string"},"operator":{"type":"string"},"values":{"items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"}},"required":["name"],"type":"object"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["describedObject","metric","target"],"type":"object"},"pods":{"properties":{"metric":{"properties":{"name":{"type":"string"},"selector":{"properties":{"matchExpressions":{"items":{"properties":{"key":{"type":"string"},"operator":{"type":"string"},"values":{"items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"}},"required":["name"],"type":"object"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["metric","target"],"type":"object"},"resource":{"properties":{"name":{"type":"string"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["name","target"],"type":"object"},"type":{"type":"string"}},"required":["type"],"type":"object"},"type":"array"},"minReplicas":{"description":"`minReplicas` is the lower limit for the number of replicas to which the autoscaler\ncan scale down. It defaults to 1 pod. minReplicas is allowed to be 0 if the\nalpha feature gate HPAScaleToZero is enabled and at least one Object or External\nmetric is configured. Scaling is active as long as at least one metric value is\navailable.","format":"int32","type":"integer"},"status":{"default":"Disabled","description":"`status` describes the desired status regarding deploying an horizontal pod autoscaler.\u003cbr\u003e\n- `Disabled` does not deploy an horizontal pod autoscaler.\u003cbr\u003e\n- `Enabled` deploys an horizontal pod autoscaler.\u003cbr\u003e","enum":["Disabled","Enabled"],"type":"string"}},"type":"object"},"enable":{"default":true,"description":"Enables the console plugin deployment.","type":"boolean"},"imagePullPolicy":{"default":"IfNotPresent","description":"`imagePullPolicy` is the Kubernetes pull policy for the image defined above","enum":["IfNotPresent","Always","Never"],"type":"string"},"logLevel":{"default":"info","description":"`logLevel` for the console plugin backend","enum":["trace","debug","info","warn","error","fatal","panic"],"type":"string"},"portNaming":{"default":{"enable":true},"description":"`portNaming` defines the configuration of the port-to-service name translation","properties":{"enable":{"default":true,"description":"Enable the console plugin port-to-service name translation","type":"boolean"},"portNames":{"additionalProperties":{"type":"string"},"description":"`portNames` defines additional port names to use in the console,\nfor example, `portNames: {\"3100\": \"loki\"}`.","type":"object"}},"type":"object"},"quickFilters":{"default":[{"default":true,"filter":{"flow_layer":"\"app\""},"name":"Applications"},{"filter":{"flow_layer":"\"infra\""},"name":"Infrastructure"},{"default":true,"filter":{"dst_kind":"\"Pod\"","src_kind":"\"Pod\""},"name":"Pods network"},{"filter":{"dst_kind":"\"Service\""},"name":"Services network"}],"description":"`quickFilters` configures quick filter presets for the Console plugin","items":{"description":"`QuickFilter` defines preset configuration for Console's quick filters","properties":{"default":{"description":"`default` defines whether this filter should be active by default or not","type":"boolean"},"filter":{"additionalProperties":{"type":"string"},"description":"`filter` is a set of keys and values to be set when this filter is selected. Each key can relate to a list of values using a coma-separated string,\nfor example, `filter: {\"src_namespace\": \"namespace1,namespace2\"}`.","type":"object"},"name":{"description":"Name of the filter, that is displayed in the Console","type":"string"}},"required":["filter","name"],"type":"object"},"type":"array"},"replicas":{"default":1,"description":"`replicas` defines the number of replicas (pods) to start.","format":"int32","minimum":0,"type":"integer"},"resources":{"default":{"limits":{"memory":"100Mi"},"requests":{"cpu":"100m","memory":"50Mi"}},"description":"`resources`, in terms of compute resources, required by this container.\nFor more information, see https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","properties":{"claims":{"description":"Claims lists the names of resources, defined in spec.resourceClaims,\nthat are used by this container.\n\nThis field depends on the\nDynamicResourceAllocation feature gate.\n\nThis field is immutable. It can only be set for containers.","items":{"description":"ResourceClaim references one entry in PodSpec.ResourceClaims.","properties":{"name":{"description":"Name must match the name of one entry in pod.spec.resourceClaims of\nthe Pod where this field is used. It makes that resource available\ninside a container.","type":"string"},"request":{"description":"Request is the name chosen for a request in the referenced claim.\nIf empty, everything from the claim is made available, otherwise\nonly the result of this request.","type":"string"}},"required":["name"],"type":"object"},"type":"array","x-kubernetes-list-map-keys":["name"],"x-kubernetes-list-type":"map"},"limits":{"additionalProperties":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"description":"Limits describes the maximum amount of compute resources allowed.\nMore info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","type":"object"},"requests":{"additionalProperties":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"description":"Requests describes the minimum amount of compute resources required.\nIf Requests is omitted for a container, it defaults to Limits if that is explicitly specified,\notherwise to an implementation-defined value. Requests cannot exceed Limits.\nMore info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","type":"object"}},"type":"object"}},"type":"object"},"deploymentModel":{"default":"Direct","description":"`deploymentModel` defines the desired type of deployment for flow processing. Possible values are:\u003cbr\u003e\n- `Direct` (default) to make the flow processor listen directly from the agents.\u003cbr\u003e\n- `Kafka` to make flows sent to a Kafka pipeline before consumption by the processor.\u003cbr\u003e\nKafka can provide better scalability, resiliency, and high availability (for more details, see https://www.redhat.com/en/topics/integration/what-is-apache-kafka).","enum":["Direct","Kafka"],"type":"string"},"exporters":{"description":"`exporters` defines additional optional exporters for custom consumption or storage.","items":{"description":"`FlowCollectorExporter` defines an additional exporter to send enriched flows to.","properties":{"ipfix":{"description":"IPFIX configuration, such as the IP address and port to send enriched IPFIX flows to.","properties":{"targetHost":{"default":"","description":"Address of the IPFIX external receiver.","type":"string"},"targetPort":{"default":4739,"description":"Port for the IPFIX external receiver.","type":"integer"},"transport":{"description":"Transport protocol (`TCP` or `UDP`) to be used for the IPFIX connection, defaults to `TCP`.","enum":["TCP","UDP"],"type":"string"}},"required":["targetHost","targetPort"],"type":"object"},"kafka":{"description":"Kafka configuration, such as the address and topic, to send enriched flows to.","properties":{"address":{"default":"","description":"Address of the Kafka server","type":"string"},"sasl":{"description":"SASL authentication configuration. [Unsupported (*)].","properties":{"clientIDReference":{"description":"Reference to the secret or config map containing the client ID","properties":{"file":{"description":"File name within the config map or secret.","type":"string"},"name":{"description":"Name of the config map or secret containing the file.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing the file. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the file reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"clientSecretReference":{"description":"Reference to the secret or config map containing the client secret","properties":{"file":{"description":"File name within the config map or secret.","type":"string"},"name":{"description":"Name of the config map or secret containing the file.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing the file. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the file reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"type":{"default":"Disabled","description":"Type of SASL authentication to use, or `Disabled` if SASL is not used","enum":["Disabled","Plain","ScramSHA512"],"type":"string"}},"type":"object"},"tls":{"description":"TLS client configuration. When using TLS, verify that the address matches the Kafka port used for TLS, generally 9093.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"},"topic":{"default":"","description":"Kafka topic to use. It must exist. NetObserv does not create it.","type":"string"}},"required":["address","topic"],"type":"object"},"openTelemetry":{"description":"OpenTelemetry configuration, such as the IP address and port to send enriched logs or metrics to.","properties":{"fieldsMapping":{"description":"Custom fields mapping to an OpenTelemetry conformant format.\nBy default, NetObserv format proposal is used: https://github.com/rhobs/observability-data-model/blob/main/network-observability.md#format-proposal .\nAs there is currently no accepted standard for L3 or L4 enriched network logs, you can freely override it with your own.","items":{"properties":{"input":{"type":"string"},"multiplier":{"type":"integer"},"output":{"type":"string"}},"type":"object"},"type":"array"},"headers":{"additionalProperties":{"type":"string"},"description":"Headers to add to messages (optional)","type":"object"},"logs":{"description":"OpenTelemetry configuration for logs.","properties":{"enable":{"default":true,"description":"Set `enable` to `true` to send logs to an OpenTelemetry receiver.","type":"boolean"}},"type":"object"},"metrics":{"description":"OpenTelemetry configuration for metrics.","properties":{"enable":{"default":true,"description":"Set `enable` to `true` to send metrics to an OpenTelemetry receiver.","type":"boolean"},"pushTimeInterval":{"default":"20s","description":"Specify how often metrics are sent to a collector.","type":"string"}},"type":"object"},"protocol":{"description":"Protocol of the OpenTelemetry connection. The available options are `http` and `grpc`.","enum":["http","grpc"],"type":"string"},"targetHost":{"default":"","description":"Address of the OpenTelemetry receiver.","type":"string"},"targetPort":{"default":4317,"description":"Port for the OpenTelemetry receiver.","type":"integer"},"tls":{"description":"TLS client configuration.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"}},"required":["targetHost","targetPort"],"type":"object"},"type":{"description":"`type` selects the type of exporters. The available options are `Kafka`, `IPFIX`, and `OpenTelemetry`.","enum":["Kafka","IPFIX","OpenTelemetry"],"type":"string"}},"required":["type"],"type":"object"},"type":"array"},"kafka":{"description":"Kafka configuration, allowing to use Kafka as a broker as part of the flow collection pipeline. Available when the `spec.deploymentModel` is `Kafka`.","properties":{"address":{"default":"","description":"Address of the Kafka server","type":"string"},"sasl":{"description":"SASL authentication configuration. [Unsupported (*)].","properties":{"clientIDReference":{"description":"Reference to the secret or config map containing the client ID","properties":{"file":{"description":"File name within the config map or secret.","type":"string"},"name":{"description":"Name of the config map or secret containing the file.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing the file. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the file reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"clientSecretReference":{"description":"Reference to the secret or config map containing the client secret","properties":{"file":{"description":"File name within the config map or secret.","type":"string"},"name":{"description":"Name of the config map or secret containing the file.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing the file. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the file reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"type":{"default":"Disabled","description":"Type of SASL authentication to use, or `Disabled` if SASL is not used","enum":["Disabled","Plain","ScramSHA512"],"type":"string"}},"type":"object"},"tls":{"description":"TLS client configuration. When using TLS, verify that the address matches the Kafka port used for TLS, generally 9093.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"},"topic":{"default":"","description":"Kafka topic to use. It must exist. NetObserv does not create it.","type":"string"}},"required":["address","topic"],"type":"object"},"loki":{"description":"`loki`, the flow store, client settings.","properties":{"advanced":{"description":"`advanced` allows setting some aspects of the internal configuration of the Loki clients.\nThis section is aimed mostly for debugging and fine-grained performance optimizations.","properties":{"excludeLabels":{"description":"`excludeLabels` is a list of field to be excluded from the list of Loki labels.","items":{"type":"string"},"type":"array"},"staticLabels":{"additionalProperties":{"type":"string"},"default":{"app":"netobserv-flowcollector"},"description":"`staticLabels` is a map of common labels to set on each flow in Loki storage.","type":"object"},"writeMaxBackoff":{"default":"5s","description":"`writeMaxBackoff` is the maximum backoff time for Loki client connection between retries.","type":"string"},"writeMaxRetries":{"default":2,"description":"`writeMaxRetries` is the maximum number of retries for Loki client connections.","format":"int32","minimum":0,"type":"integer"},"writeMinBackoff":{"default":"1s","description":"`writeMinBackoff` is the initial backoff time for Loki client connection between retries.","type":"string"}},"type":"object"},"enable":{"default":true,"description":"Set `enable` to `true` to store flows in Loki.\nThe Console plugin can use either Loki or Prometheus as a data source for metrics (see also `spec.prometheus.querier`), or both.\nNot all queries are transposable from Loki to Prometheus. Hence, if Loki is disabled, some features of the plugin are disabled as well,\nsuch as getting per-pod information or viewing raw flows.\nIf both Prometheus and Loki are enabled, Prometheus takes precedence and Loki is used as a fallback for queries that Prometheus cannot handle.\nIf they are both disabled, the Console plugin is not deployed.","type":"boolean"},"lokiStack":{"description":"Loki configuration for `LokiStack` mode. This is useful for an easy Loki Operator configuration.\nIt is ignored for other modes.","properties":{"name":{"default":"loki","description":"Name of an existing LokiStack resource to use.","type":"string"},"namespace":{"description":"Namespace where this `LokiStack` resource is located. If omitted, it is assumed to be the same as `spec.namespace`.","type":"string"}},"required":["name"],"type":"object"},"manual":{"description":"Loki configuration for `Manual` mode. This is the most flexible configuration.\nIt is ignored for other modes.","properties":{"authToken":{"default":"Disabled","description":"`authToken` describes the way to get a token to authenticate to Loki.\u003cbr\u003e\n- `Disabled` does not send any token with the request.\u003cbr\u003e\n- `Forward` forwards the user token for authorization.\u003cbr\u003e\n- `Host` [deprecated (*)] - uses the local pod service account to authenticate to Loki.\u003cbr\u003e\nWhen using the Loki Operator, this must be set to `Forward`.","enum":["Disabled","Host","Forward"],"type":"string"},"ingesterUrl":{"default":"http://loki:3100/","description":"`ingesterUrl` is the address of an existing Loki ingester service to push the flows to. When using the Loki Operator,\nset it to the Loki gateway service with the `network` tenant set in path, for example\nhttps://loki-gateway-http.netobserv.svc:8080/api/logs/v1/network.","type":"string"},"querierUrl":{"default":"http://loki:3100/","description":"`querierUrl` specifies the address of the Loki querier service.\nWhen using the Loki Operator, set it to the Loki gateway service with the `network` tenant set in path, for example\nhttps://loki-gateway-http.netobserv.svc:8080/api/logs/v1/network.","type":"string"},"statusTls":{"description":"TLS client configuration for Loki status URL.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"},"statusUrl":{"description":"`statusUrl` specifies the address of the Loki `/ready`, `/metrics` and `/config` endpoints, in case it is different from the\nLoki querier URL. If empty, the `querierUrl` value is used.\nThis is useful to show error messages and some context in the frontend.\nWhen using the Loki Operator, set it to the Loki HTTP query frontend service, for example\nhttps://loki-query-frontend-http.netobserv.svc:3100/.\n`statusTLS` configuration is used when `statusUrl` is set.","type":"string"},"tenantID":{"default":"netobserv","description":"`tenantID` is the Loki `X-Scope-OrgID` that identifies the tenant for each request.\nWhen using the Loki Operator, set it to `network`, which corresponds to a special tenant mode.","type":"string"},"tls":{"description":"TLS client configuration for Loki URL.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"}},"type":"object"},"microservices":{"description":"Loki configuration for `Microservices` mode.\nUse this option when Loki is installed using the microservices deployment mode (https://grafana.com/docs/loki/latest/fundamentals/architecture/deployment-modes/#microservices-mode).\nIt is ignored for other modes.","properties":{"ingesterUrl":{"default":"http://loki-distributor:3100/","description":"`ingesterUrl` is the address of an existing Loki ingester service to push the flows to.","type":"string"},"querierUrl":{"default":"http://loki-query-frontend:3100/","description":"`querierURL` specifies the address of the Loki querier service.","type":"string"},"tenantID":{"default":"netobserv","description":"`tenantID` is the Loki `X-Scope-OrgID` header that identifies the tenant for each request.","type":"string"},"tls":{"description":"TLS client configuration for Loki URL.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"}},"type":"object"},"mode":{"default":"Monolithic","description":"`mode` must be set according to the installation mode of Loki:\u003cbr\u003e\n- Use `LokiStack` when Loki is managed using the Loki Operator\u003cbr\u003e\n- Use `Monolithic` when Loki is installed as a monolithic workload\u003cbr\u003e\n- Use `Microservices` when Loki is installed as microservices, but without Loki Operator\u003cbr\u003e\n- Use `Manual` if none of the options above match your setup\u003cbr\u003e","enum":["Manual","LokiStack","Monolithic","Microservices"],"type":"string"},"monolithic":{"description":"Loki configuration for `Monolithic` mode.\nUse this option when Loki is installed using the monolithic deployment mode (https://grafana.com/docs/loki/latest/fundamentals/architecture/deployment-modes/#monolithic-mode).\nIt is ignored for other modes.","properties":{"tenantID":{"default":"netobserv","description":"`tenantID` is the Loki `X-Scope-OrgID` header that identifies the tenant for each request.","type":"string"},"tls":{"description":"TLS client configuration for Loki URL.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"},"url":{"default":"http://loki:3100/","description":"`url` is the unique address of an existing Loki service that points to both the ingester and the querier.","type":"string"}},"type":"object"},"readTimeout":{"default":"30s","description":"`readTimeout` is the maximum console plugin loki query total time limit.\nA timeout of zero means no timeout.","type":"string"},"writeBatchSize":{"default":10485760,"description":"`writeBatchSize` is the maximum batch size (in bytes) of Loki logs to accumulate before sending.","format":"int64","minimum":1,"type":"integer"},"writeBatchWait":{"default":"1s","description":"`writeBatchWait` is the maximum time to wait before sending a Loki batch.","type":"string"},"writeTimeout":{"default":"10s","description":"`writeTimeout` is the maximum Loki time connection / request limit.\nA timeout of zero means no timeout.","type":"string"}},"required":["mode"],"type":"object"},"namespace":{"default":"netobserv","description":"Namespace where NetObserv pods are deployed.","type":"string","x-kubernetes-validations":[{"message":"Namespace is immutable. If you need to change it, delete and recreate the resource.","rule":"self == oldSelf"}]},"networkPolicy":{"description":"`networkPolicy` defines ingress network policy settings for NetObserv components isolation.","properties":{"additionalNamespaces":{"description":"`additionalNamespaces` contains additional namespaces allowed to connect to the NetObserv namespace.\nIt provides flexibility in the network policy configuration, but if you need a more specific\nconfiguration, you can disable it and install your own instead.","items":{"type":"string"},"type":"array"},"enable":{"description":"Set `enable` to `true` to deploy network policies on the namespaces used by NetObserv (main and privileged). It is disabled by default.\nThese network policies better isolate the NetObserv components to prevent undesired connections to them.\nTo increase the security of connections, enable this option or create your own network policy.","type":"boolean"}},"type":"object"},"processor":{"description":"`processor` defines the settings of the component that receives the flows from the agent,\nenriches them, generates metrics, and forwards them to the Loki persistence layer and/or any available exporter.","properties":{"addZone":{"description":"`addZone` allows availability zone awareness by labelling flows with their source and destination zones.\nThis feature requires the \"topology.kubernetes.io/zone\" label to be set on nodes.","type":"boolean"},"advanced":{"description":"`advanced` allows setting some aspects of the internal configuration of the flow processor.\nThis section is aimed mostly for debugging and fine-grained performance optimizations,\nsuch as `GOGC` and `GOMAXPROCS` environment variables. Set these values at your own risk.","properties":{"conversationEndTimeout":{"default":"10s","description":"`conversationEndTimeout` is the time to wait after a network flow is received, to consider the conversation ended.\nThis delay is ignored when a FIN packet is collected for TCP flows (see `conversationTerminatingTimeout` instead).","type":"string"},"conversationHeartbeatInterval":{"default":"30s","description":"`conversationHeartbeatInterval` is the time to wait between \"tick\" events of a conversation","type":"string"},"conversationTerminatingTimeout":{"default":"5s","description":"`conversationTerminatingTimeout` is the time to wait from detected FIN flag to end a conversation. Only relevant for TCP flows.","type":"string"},"dropUnusedFields":{"default":true,"description":"`dropUnusedFields` [deprecated (*)] this setting is not used anymore.","type":"boolean"},"enableKubeProbes":{"default":true,"description":"`enableKubeProbes` is a flag to enable or disable Kubernetes liveness and readiness probes","type":"boolean"},"env":{"additionalProperties":{"type":"string"},"description":"`env` allows passing custom environment variables to underlying components. Useful for passing\nsome very concrete performance-tuning options, such as `GOGC` and `GOMAXPROCS`, that should not be\npublicly exposed as part of the FlowCollector descriptor, as they are only useful\nin edge debug or support scenarios.","type":"object"},"healthPort":{"default":8080,"description":"`healthPort` is a collector HTTP port in the Pod that exposes the health check API","format":"int32","maximum":65535,"minimum":1,"type":"integer"},"port":{"default":2055,"description":"Port of the flow collector (host port).\nBy convention, some values are forbidden. It must be greater than 1024 and different from\n4500, 4789 and 6081.","format":"int32","maximum":65535,"minimum":1025,"type":"integer"},"profilePort":{"description":"`profilePort` allows setting up a Go pprof profiler listening to this port","format":"int32","maximum":65535,"minimum":0,"type":"integer"},"scheduling":{"description":"scheduling controls how the pods are scheduled on nodes.","properties":{"affinity":{"description":"If specified, the pod's scheduling constraints. For documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#scheduling.","properties":{"nodeAffinity":{"description":"Describes node affinity scheduling rules for the pod.","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and adding\n\"weight\" to the sum if the node matches the corresponding matchExpressions; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"An empty preferred scheduling term matches all objects with implicit weight 0\n(i.e. it's a no-op). A null preferred scheduling term matches no objects (i.e. is also a no-op).","properties":{"preference":{"description":"A node selector term, associated with the corresponding weight.","properties":{"matchExpressions":{"description":"A list of node selector requirements by node's labels.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchFields":{"description":"A list of node selector requirements by node's fields.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object","x-kubernetes-map-type":"atomic"},"weight":{"description":"Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100.","format":"int32","type":"integer"}},"required":["preference","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to an update), the system\nmay or may not try to eventually evict the pod from its node.","properties":{"nodeSelectorTerms":{"description":"Required. A list of node selector terms. The terms are ORed.","items":{"description":"A null or empty node selector term matches no objects. The requirements of\nthem are ANDed.\nThe TopologySelectorTerm type implements a subset of the NodeSelectorTerm.","properties":{"matchExpressions":{"description":"A list of node selector requirements by node's labels.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchFields":{"description":"A list of node selector requirements by node's fields.","items":{"description":"A node selector requirement is a selector that contains values, a key, and an operator\nthat relates the key and values.","properties":{"key":{"description":"The label key that the selector applies to.","type":"string"},"operator":{"description":"Represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists, DoesNotExist. Gt, and Lt.","type":"string"},"values":{"description":"An array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. If the operator is Gt or Lt, the values\narray must have a single element, which will be interpreted as an integer.\nThis array is replaced during a strategic merge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object","x-kubernetes-map-type":"atomic"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["nodeSelectorTerms"],"type":"object","x-kubernetes-map-type":"atomic"}},"type":"object"},"podAffinity":{"description":"Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)).","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and adding\n\"weight\" to the sum if the node has pods which matches the corresponding podAffinityTerm; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s)","properties":{"podAffinityTerm":{"description":"Required. A pod affinity term, associated with the corresponding weight.","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"weight":{"description":"weight associated with matching the corresponding podAffinityTerm,\nin the range 1-100.","format":"int32","type":"integer"}},"required":["podAffinityTerm","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to a pod label update), the\nsystem may or may not try to eventually evict the pod from its node.\nWhen there are multiple elements, the lists of nodes corresponding to each\npodAffinityTerm are intersected, i.e. all terms must be satisfied.","items":{"description":"Defines a set of pods (namely those matching the labelSelector\nrelative to the given namespace(s)) that this pod should be\nco-located (affinity) or not co-located (anti-affinity) with,\nwhere co-located is defined as running on a node whose value of\nthe label with key \u003ctopologyKey\u003e matches that of any node on which\na pod of the set of pods is running","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object"},"podAntiAffinity":{"description":"Describes pod anti-affinity scheduling rules (e.g. avoid putting this pod in the same node, zone, etc. as some other pod(s)).","properties":{"preferredDuringSchedulingIgnoredDuringExecution":{"description":"The scheduler will prefer to schedule pods to nodes that satisfy\nthe anti-affinity expressions specified by this field, but it may choose\na node that violates one or more of the expressions. The node that is\nmost preferred is the one with the greatest sum of weights, i.e.\nfor each node that meets all of the scheduling requirements (resource\nrequest, requiredDuringScheduling anti-affinity expressions, etc.),\ncompute a sum by iterating through the elements of this field and subtracting\n\"weight\" from the sum if the node has pods which matches the corresponding podAffinityTerm; the\nnode(s) with the highest sum are the most preferred.","items":{"description":"The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s)","properties":{"podAffinityTerm":{"description":"Required. A pod affinity term, associated with the corresponding weight.","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"weight":{"description":"weight associated with matching the corresponding podAffinityTerm,\nin the range 1-100.","format":"int32","type":"integer"}},"required":["podAffinityTerm","weight"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"requiredDuringSchedulingIgnoredDuringExecution":{"description":"If the anti-affinity requirements specified by this field are not met at\nscheduling time, the pod will not be scheduled onto the node.\nIf the anti-affinity requirements specified by this field cease to be met\nat some point during pod execution (e.g. due to a pod label update), the\nsystem may or may not try to eventually evict the pod from its node.\nWhen there are multiple elements, the lists of nodes corresponding to each\npodAffinityTerm are intersected, i.e. all terms must be satisfied.","items":{"description":"Defines a set of pods (namely those matching the labelSelector\nrelative to the given namespace(s)) that this pod should be\nco-located (affinity) or not co-located (anti-affinity) with,\nwhere co-located is defined as running on a node whose value of\nthe label with key \u003ctopologyKey\u003e matches that of any node on which\na pod of the set of pods is running","properties":{"labelSelector":{"description":"A label query over a set of resources, in this case pods.\nIf it's null, this PodAffinityTerm matches with no Pods.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"matchLabelKeys":{"description":"MatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key in (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both matchLabelKeys and labelSelector.\nAlso, matchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"mismatchLabelKeys":{"description":"MismatchLabelKeys is a set of pod label keys to select which pods will\nbe taken into consideration. The keys are used to lookup values from the\nincoming pod labels, those key-value labels are merged with `labelSelector` as `key notin (value)`\nto select the group of existing pods which pods will be taken into consideration\nfor the incoming pod's pod (anti) affinity. Keys that don't exist in the incoming\npod labels will be ignored. The default value is empty.\nThe same key is forbidden to exist in both mismatchLabelKeys and labelSelector.\nAlso, mismatchLabelKeys cannot be set when labelSelector isn't set.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"namespaceSelector":{"description":"A label query over the set of namespaces that the term applies to.\nThe term is applied to the union of the namespaces selected by this field\nand the ones listed in the namespaces field.\nnull selector and null or empty namespaces list means \"this pod's namespace\".\nAn empty selector ({}) matches all namespaces.","properties":{"matchExpressions":{"description":"matchExpressions is a list of label selector requirements. The requirements are ANDed.","items":{"description":"A label selector requirement is a selector that contains values, a key, and an operator that\nrelates the key and values.","properties":{"key":{"description":"key is the label key that the selector applies to.","type":"string"},"operator":{"description":"operator represents a key's relationship to a set of values.\nValid operators are In, NotIn, Exists and DoesNotExist.","type":"string"},"values":{"description":"values is an array of string values. If the operator is In or NotIn,\nthe values array must be non-empty. If the operator is Exists or DoesNotExist,\nthe values array must be empty. This array is replaced during a strategic\nmerge patch.","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"description":"matchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels\nmap is equivalent to an element of matchExpressions, whose key field is \"key\", the\noperator is \"In\", and the values array contains only \"value\". The requirements are ANDed.","type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"},"namespaces":{"description":"namespaces specifies a static list of namespace names that the term applies to.\nThe term is applied to the union of the namespaces listed in this field\nand the ones selected by namespaceSelector.\nnull or empty namespaces list and null namespaceSelector means \"this pod's namespace\".","items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"},"topologyKey":{"description":"This pod should be co-located (affinity) or not co-located (anti-affinity) with the pods matching\nthe labelSelector in the specified namespaces, where co-located is defined as running on a node\nwhose value of the label with key topologyKey matches that of any node on which any of the\nselected pods is running.\nEmpty topologyKey is not allowed.","type":"string"}},"required":["topologyKey"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"}},"type":"object"}},"type":"object"},"nodeSelector":{"additionalProperties":{"type":"string"},"description":"`nodeSelector` allows scheduling of pods only onto nodes that have each of the specified labels.\nFor documentation, refer to https://kubernetes.io/docs/concepts/configuration/assign-pod-node/.","type":"object","x-kubernetes-map-type":"atomic"},"priorityClassName":{"description":"If specified, indicates the pod's priority. For documentation, refer to https://kubernetes.io/docs/concepts/scheduling-eviction/pod-priority-preemption/#how-to-use-priority-and-preemption.\nIf not specified, default priority is used, or zero if there is no default.","type":"string"},"tolerations":{"description":"`tolerations` is a list of tolerations that allow the pod to schedule onto nodes with matching taints.\nFor documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/pod-v1/#scheduling.","items":{"description":"The pod this Toleration is attached to tolerates any taint that matches\nthe triple \u003ckey,value,effect\u003e using the matching operator \u003coperator\u003e.","properties":{"effect":{"description":"Effect indicates the taint effect to match. Empty means match all taint effects.\nWhen specified, allowed values are NoSchedule, PreferNoSchedule and NoExecute.","type":"string"},"key":{"description":"Key is the taint key that the toleration applies to. Empty means match all taint keys.\nIf the key is empty, operator must be Exists; this combination means to match all values and all keys.","type":"string"},"operator":{"description":"Operator represents a key's relationship to the value.\nValid operators are Exists and Equal. Defaults to Equal.\nExists is equivalent to wildcard for value, so that a pod can\ntolerate all taints of a particular category.","type":"string"},"tolerationSeconds":{"description":"TolerationSeconds represents the period of time the toleration (which must be\nof effect NoExecute, otherwise this field is ignored) tolerates the taint. By default,\nit is not set, which means tolerate the taint forever (do not evict). Zero and\nnegative values will be treated as 0 (evict immediately) by the system.","format":"int64","type":"integer"},"value":{"description":"Value is the taint value the toleration matches to.\nIf the operator is Exists, the value should be empty, otherwise just a regular string.","type":"string"}},"type":"object"},"type":"array"}},"type":"object"},"secondaryNetworks":{"description":"Defines secondary networks to be checked for resources identification.\nTo guarantee a correct identification, indexed values must form an unique identifier across the cluster.\nIf the same index is used by several resources, those resources might be incorrectly labeled.","items":{"properties":{"index":{"description":"`index` is a list of fields to use for indexing the pods. They should form a unique Pod identifier across the cluster.\nCan be any of: `MAC`, `IP`, `Interface`.\nFields absent from the 'k8s.v1.cni.cncf.io/network-status' annotation must not be added to the index.","items":{"description":"Field to index for secondary network pod identification, can be any of: `MAC`, `IP`, `Interface`.","enum":["MAC","IP","Interface"],"type":"string"},"type":"array"},"name":{"description":"`name` should match the network name as visible in the pods annotation 'k8s.v1.cni.cncf.io/network-status'.","type":"string"}},"required":["index","name"],"type":"object"},"type":"array"}},"type":"object"},"clusterName":{"default":"","description":"`clusterName` is the name of the cluster to appear in the flows data. This is useful in a multi-cluster context. When using OpenShift, leave empty to make it automatically determined.","type":"string"},"deduper":{"description":"`deduper` allows you to sample or drop flows identified as duplicates, in order to save on resource usage.","properties":{"mode":{"default":"Disabled","description":"Set the Processor de-duplication mode. It comes in addition to the Agent-based deduplication, since the Agent cannot de-duplicate same flows reported from different nodes.\u003cbr\u003e\n- Use `Drop` to drop every flow considered as duplicates, allowing saving more on resource usage but potentially losing some information such as the network interfaces used from peer, or network events.\u003cbr\u003e\n- Use `Sample` to randomly keep only one flow on 50, which is the default, among the ones considered as duplicates. This is a compromise between dropping every duplicate or keeping every duplicate. This sampling action comes in addition to the Agent-based sampling. If both Agent and Processor sampling values are `50`, the combined sampling is 1:2500.\u003cbr\u003e\n- Use `Disabled` to turn off Processor-based de-duplication.\u003cbr\u003e","enum":["Disabled","Drop","Sample"],"type":"string"},"sampling":{"default":50,"description":"`sampling` is the sampling ratio when deduper `mode` is `Sample`. For example, a value of `50` means that 1 flow in 50 is sampled.","format":"int32","minimum":0,"type":"integer"}},"type":"object"},"filters":{"description":"`filters` lets you define custom filters to limit the amount of generated flows.\nThese filters provide more flexibility than the eBPF Agent filters (in `spec.agent.ebpf.flowFilter`), such as allowing to filter by Kubernetes namespace,\nbut with a lesser improvement in performance.","items":{"description":"`FLPFilterSet` defines the desired configuration for FLP-based filtering satisfying all conditions.","properties":{"outputTarget":{"description":"If specified, these filters target a single output: `Loki`, `Metrics` or `Exporters`. By default, all outputs are targeted.","enum":["","Loki","Metrics","Exporters"],"type":"string"},"query":{"description":"A query that selects the network flows to keep. More information about this query language in https://github.com/netobserv/flowlogs-pipeline/blob/main/docs/filtering.md.","type":"string"},"sampling":{"description":"`sampling` is an optional sampling ratio to apply to this filter. For example, a value of `50` means that 1 matching flow in 50 is sampled.","format":"int32","minimum":0,"type":"integer"}},"type":"object"},"type":"array"},"imagePullPolicy":{"default":"IfNotPresent","description":"`imagePullPolicy` is the Kubernetes pull policy for the image defined above","enum":["IfNotPresent","Always","Never"],"type":"string"},"kafkaConsumerAutoscaler":{"description":"`kafkaConsumerAutoscaler` is the spec of a horizontal pod autoscaler to set up for `flowlogs-pipeline-transformer`, which consumes Kafka messages.\nThis setting is ignored when Kafka is disabled.","properties":{"maxReplicas":{"default":3,"description":"`maxReplicas` is the upper limit for the number of pods that can be set by the autoscaler; cannot be smaller than MinReplicas.","format":"int32","type":"integer"},"metrics":{"description":"Metrics used by the pod autoscaler. For documentation, refer to https://kubernetes.io/docs/reference/kubernetes-api/workload-resources/horizontal-pod-autoscaler-v2/","items":{"properties":{"containerResource":{"properties":{"container":{"type":"string"},"name":{"type":"string"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["container","name","target"],"type":"object"},"external":{"properties":{"metric":{"properties":{"name":{"type":"string"},"selector":{"properties":{"matchExpressions":{"items":{"properties":{"key":{"type":"string"},"operator":{"type":"string"},"values":{"items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"}},"required":["name"],"type":"object"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["metric","target"],"type":"object"},"object":{"properties":{"describedObject":{"properties":{"apiVersion":{"type":"string"},"kind":{"type":"string"},"name":{"type":"string"}},"required":["kind","name"],"type":"object"},"metric":{"properties":{"name":{"type":"string"},"selector":{"properties":{"matchExpressions":{"items":{"properties":{"key":{"type":"string"},"operator":{"type":"string"},"values":{"items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"}},"required":["name"],"type":"object"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["describedObject","metric","target"],"type":"object"},"pods":{"properties":{"metric":{"properties":{"name":{"type":"string"},"selector":{"properties":{"matchExpressions":{"items":{"properties":{"key":{"type":"string"},"operator":{"type":"string"},"values":{"items":{"type":"string"},"type":"array","x-kubernetes-list-type":"atomic"}},"required":["key","operator"],"type":"object"},"type":"array","x-kubernetes-list-type":"atomic"},"matchLabels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object","x-kubernetes-map-type":"atomic"}},"required":["name"],"type":"object"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["metric","target"],"type":"object"},"resource":{"properties":{"name":{"type":"string"},"target":{"properties":{"averageUtilization":{"format":"int32","type":"integer"},"averageValue":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"type":{"type":"string"},"value":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true}},"required":["type"],"type":"object"}},"required":["name","target"],"type":"object"},"type":{"type":"string"}},"required":["type"],"type":"object"},"type":"array"},"minReplicas":{"description":"`minReplicas` is the lower limit for the number of replicas to which the autoscaler\ncan scale down. It defaults to 1 pod. minReplicas is allowed to be 0 if the\nalpha feature gate HPAScaleToZero is enabled and at least one Object or External\nmetric is configured. Scaling is active as long as at least one metric value is\navailable.","format":"int32","type":"integer"},"status":{"default":"Disabled","description":"`status` describes the desired status regarding deploying an horizontal pod autoscaler.\u003cbr\u003e\n- `Disabled` does not deploy an horizontal pod autoscaler.\u003cbr\u003e\n- `Enabled` deploys an horizontal pod autoscaler.\u003cbr\u003e","enum":["Disabled","Enabled"],"type":"string"}},"type":"object"},"kafkaConsumerBatchSize":{"default":10485760,"description":"`kafkaConsumerBatchSize` indicates to the broker the maximum batch size, in bytes, that the consumer accepts. Ignored when not using Kafka. Default: 10MB.","type":"integer"},"kafkaConsumerQueueCapacity":{"default":1000,"description":"`kafkaConsumerQueueCapacity` defines the capacity of the internal message queue used in the Kafka consumer client. Ignored when not using Kafka.","type":"integer"},"kafkaConsumerReplicas":{"default":3,"description":"`kafkaConsumerReplicas` defines the number of replicas (pods) to start for `flowlogs-pipeline-transformer`, which consumes Kafka messages.\nThis setting is ignored when Kafka is disabled.","format":"int32","minimum":0,"type":"integer"},"logLevel":{"default":"info","description":"`logLevel` of the processor runtime","enum":["trace","debug","info","warn","error","fatal","panic"],"type":"string"},"logTypes":{"default":"Flows","description":"`logTypes` defines the desired record types to generate. Possible values are:\u003cbr\u003e\n- `Flows` to export regular network flows. This is the default.\u003cbr\u003e\n- `Conversations` to generate events for started conversations, ended conversations as well as periodic \"tick\" updates. Note that in this mode, Prometheus metrics are not accurate on long-standing conversations.\u003cbr\u003e\n- `EndedConversations` to generate only ended conversations events. Note that in this mode, Prometheus metrics are not accurate on long-standing conversations.\u003cbr\u003e\n- `All` to generate both network flows and all conversations events. It is not recommended due to the impact on resources footprint.\u003cbr\u003e","enum":["Flows","Conversations","EndedConversations","All"],"type":"string"},"metrics":{"description":"`Metrics` define the processor configuration regarding metrics","properties":{"disableAlerts":{"description":"`disableAlerts` is a list of alerts that should be disabled.\nPossible values are:\u003cbr\u003e\n`NetObservNoFlows`, which is triggered when no flows are being observed for a certain period.\u003cbr\u003e\n`NetObservLokiError`, which is triggered when flows are being dropped due to Loki errors.\u003cbr\u003e","items":{"description":"Name of a processor alert.\nPossible values are:\u003cbr\u003e\n- `NetObservNoFlows`, which is triggered when no flows are being observed for a certain period.\u003cbr\u003e\n- `NetObservLokiError`, which is triggered when flows are being dropped due to Loki errors.\u003cbr\u003e","enum":["NetObservNoFlows","NetObservLokiError"],"type":"string"},"type":"array"},"includeList":{"description":"`includeList` is a list of metric names to specify which ones to generate.\nThe names correspond to the names in Prometheus without the prefix. For example,\n`namespace_egress_packets_total` shows up as `netobserv_namespace_egress_packets_total` in Prometheus.\nNote that the more metrics you add, the bigger is the impact on Prometheus workload resources.\nMetrics enabled by default are:\n`namespace_flows_total`, `node_ingress_bytes_total`, `node_egress_bytes_total`, `workload_ingress_bytes_total`,\n`workload_egress_bytes_total`, `namespace_drop_packets_total` (when `PacketDrop` feature is enabled),\n`namespace_rtt_seconds` (when `FlowRTT` feature is enabled), `namespace_dns_latency_seconds` (when `DNSTracking` feature is enabled),\n`namespace_network_policy_events_total` (when `NetworkEvents` feature is enabled).\nMore information, with full list of available metrics: https://github.com/netobserv/network-observability-operator/blob/main/docs/Metrics.md","items":{"description":"Metric name. More information in https://github.com/netobserv/network-observability-operator/blob/main/docs/Metrics.md.","enum":["namespace_egress_bytes_total","namespace_egress_packets_total","namespace_ingress_bytes_total","namespace_ingress_packets_total","namespace_flows_total","node_egress_bytes_total","node_egress_packets_total","node_ingress_bytes_total","node_ingress_packets_total","node_flows_total","workload_egress_bytes_total","workload_egress_packets_total","workload_ingress_bytes_total","workload_ingress_packets_total","workload_flows_total","namespace_drop_bytes_total","namespace_drop_packets_total","node_drop_bytes_total","node_drop_packets_total","workload_drop_bytes_total","workload_drop_packets_total","namespace_rtt_seconds","node_rtt_seconds","workload_rtt_seconds","namespace_dns_latency_seconds","node_dns_latency_seconds","workload_dns_latency_seconds","node_network_policy_events_total","namespace_network_policy_events_total","workload_network_policy_events_total","node_ipsec_flows_total","node_to_node_ingress_flows_total"],"type":"string"},"type":"array"},"server":{"description":"Metrics server endpoint configuration for Prometheus scraper","properties":{"port":{"description":"The metrics server HTTP port.","format":"int32","maximum":65535,"minimum":1,"type":"integer"},"tls":{"description":"TLS configuration.","properties":{"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the provided certificate.\nIf set to `true`, the `providedCaFile` field is ignored.","type":"boolean"},"provided":{"description":"TLS configuration when `type` is set to `Provided`.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"providedCaFile":{"description":"Reference to the CA file when `type` is set to `Provided`.","properties":{"file":{"description":"File name within the config map or secret.","type":"string"},"name":{"description":"Name of the config map or secret containing the file.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing the file. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the file reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"type":{"default":"Disabled","description":"Select the type of TLS configuration:\u003cbr\u003e\n- `Disabled` (default) to not configure TLS for the endpoint.\n- `Provided` to manually provide cert file and a key file. [Unsupported (*)].\n- `Auto` to use OpenShift auto generated certificate using annotations.","enum":["Disabled","Provided","Auto"],"type":"string"}},"required":["type"],"type":"object"}},"type":"object"}},"type":"object"},"multiClusterDeployment":{"default":false,"description":"Set `multiClusterDeployment` to `true` to enable multi clusters feature. This adds `clusterName` label to flows data","type":"boolean"},"resources":{"default":{"limits":{"memory":"800Mi"},"requests":{"cpu":"100m","memory":"100Mi"}},"description":"`resources` are the compute resources required by this container.\nFor more information, see https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","properties":{"claims":{"description":"Claims lists the names of resources, defined in spec.resourceClaims,\nthat are used by this container.\n\nThis field depends on the\nDynamicResourceAllocation feature gate.\n\nThis field is immutable. It can only be set for containers.","items":{"description":"ResourceClaim references one entry in PodSpec.ResourceClaims.","properties":{"name":{"description":"Name must match the name of one entry in pod.spec.resourceClaims of\nthe Pod where this field is used. It makes that resource available\ninside a container.","type":"string"},"request":{"description":"Request is the name chosen for a request in the referenced claim.\nIf empty, everything from the claim is made available, otherwise\nonly the result of this request.","type":"string"}},"required":["name"],"type":"object"},"type":"array","x-kubernetes-list-map-keys":["name"],"x-kubernetes-list-type":"map"},"limits":{"additionalProperties":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"description":"Limits describes the maximum amount of compute resources allowed.\nMore info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","type":"object"},"requests":{"additionalProperties":{"anyOf":[{"type":"integer"},{"type":"string"}],"pattern":"^(\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\\+|-)?(([0-9]+(\\.[0-9]*)?)|(\\.[0-9]+))))?$","x-kubernetes-int-or-string":true},"description":"Requests describes the minimum amount of compute resources required.\nIf Requests is omitted for a container, it defaults to Limits if that is explicitly specified,\notherwise to an implementation-defined value. Requests cannot exceed Limits.\nMore info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/","type":"object"}},"type":"object"},"subnetLabels":{"description":"`subnetLabels` allows to define custom labels on subnets and IPs or to enable automatic labelling of recognized subnets in OpenShift, which is used to identify cluster external traffic.\nWhen a subnet matches the source or destination IP of a flow, a corresponding field is added: `SrcSubnetLabel` or `DstSubnetLabel`.","properties":{"customLabels":{"description":"`customLabels` allows to customize subnets and IPs labelling, such as to identify cluster-external workloads or web services.\nIf you enable `openShiftAutoDetect`, `customLabels` can override the detected subnets in case they overlap.","items":{"description":"SubnetLabel allows to label subnets and IPs, such as to identify cluster-external workloads or web services.","properties":{"cidrs":{"description":"List of CIDRs, such as `[\"1.2.3.4/32\"]`.","items":{"type":"string"},"type":"array"},"name":{"description":"Label name, used to flag matching flows.","type":"string"}},"required":["cidrs","name"],"type":"object"},"type":"array"},"openShiftAutoDetect":{"description":"`openShiftAutoDetect` allows, when set to `true`, to detect automatically the machines, pods and services subnets based on the\nOpenShift install configuration and the Cluster Network Operator configuration. Indirectly, this is a way to accurately detect\nexternal traffic: flows that are not labeled for those subnets are external to the cluster. Enabled by default on OpenShift.","type":"boolean"}},"type":"object"}},"type":"object"},"prometheus":{"description":"`prometheus` defines Prometheus settings, such as querier configuration used to fetch metrics from the Console plugin.","properties":{"querier":{"description":"Prometheus querying configuration, such as client settings, used in the Console plugin.","properties":{"enable":{"description":"When `enable` is `true`, the Console plugin queries flow metrics from Prometheus instead of Loki whenever possible.\nIt is enbaled by default: set it to `false` to disable this feature.\nThe Console plugin can use either Loki or Prometheus as a data source for metrics (see also `spec.loki`), or both.\nNot all queries are transposable from Loki to Prometheus. Hence, if Loki is disabled, some features of the plugin are disabled as well,\nsuch as getting per-pod information or viewing raw flows.\nIf both Prometheus and Loki are enabled, Prometheus takes precedence and Loki is used as a fallback for queries that Prometheus cannot handle.\nIf they are both disabled, the Console plugin is not deployed.","type":"boolean"},"manual":{"description":"Prometheus configuration for `Manual` mode.","properties":{"forwardUserToken":{"description":"Set `true` to forward logged in user token in queries to Prometheus","type":"boolean"},"tls":{"description":"TLS client configuration for Prometheus URL.","properties":{"caCert":{"description":"`caCert` defines the reference of the certificate for the Certificate Authority.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"},"enable":{"default":false,"description":"Enable TLS","type":"boolean"},"insecureSkipVerify":{"default":false,"description":"`insecureSkipVerify` allows skipping client-side verification of the server certificate.\nIf set to `true`, the `caCert` field is ignored.","type":"boolean"},"userCert":{"description":"`userCert` defines the user certificate reference and is used for mTLS. When you use one-way TLS, you can ignore this property.","properties":{"certFile":{"description":"`certFile` defines the path to the certificate file name within the config map or secret.","type":"string"},"certKey":{"description":"`certKey` defines the path to the certificate private key file name within the config map or secret. Omit when the key is not necessary.","type":"string"},"name":{"description":"Name of the config map or secret containing certificates.","type":"string"},"namespace":{"default":"","description":"Namespace of the config map or secret containing certificates. If omitted, the default is to use the same namespace as where NetObserv is deployed.\nIf the namespace is different, the config map or the secret is copied so that it can be mounted as required.","type":"string"},"type":{"description":"Type for the certificate reference: `configmap` or `secret`.","enum":["configmap","secret"],"type":"string"}},"type":"object"}},"type":"object"},"url":{"default":"http://prometheus:9090","description":"`url` is the address of an existing Prometheus service to use for querying metrics.","type":"string"}},"type":"object"},"mode":{"default":"Auto","description":"`mode` must be set according to the type of Prometheus installation that stores NetObserv metrics:\u003cbr\u003e\n- Use `Auto` to try configuring automatically. In OpenShift, it uses the Thanos querier from OpenShift Cluster Monitoring\u003cbr\u003e\n- Use `Manual` for a manual setup\u003cbr\u003e","enum":["Manual","Auto"],"type":"string"},"timeout":{"default":"30s","description":"`timeout` is the read timeout for console plugin queries to Prometheus.\nA timeout of zero means no timeout.","type":"string"}},"required":["mode"],"type":"object"}},"type":"object"}},"type":"object"},"status":{"description":"`FlowCollectorStatus` defines the observed state of FlowCollector","properties":{"conditions":{"description":"`conditions` represents the latest available observations of an object's state","items":{"description":"Condition contains details for one aspect of the current state of this API Resource.","properties":{"lastTransitionTime":{"description":"lastTransitionTime is the last time the condition transitioned from one status to another.\nThis should be when the underlying condition changed.  If that is not known, then using the time when the API field changed is acceptable.","format":"date-time","type":"string"},"message":{"description":"message is a human readable message indicating details about the transition.\nThis may be an empty string.","maxLength":32768,"type":"string"},"observedGeneration":{"description":"observedGeneration represents the .metadata.generation that the condition was set based upon.\nFor instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date\nwith respect to the current state of the instance.","format":"int64","minimum":0,"type":"integer"},"reason":{"description":"reason contains a programmatic identifier indicating the reason for the condition's last transition.\nProducers of specific condition types may define expected values and meanings for this field,\nand whether the values are considered a guaranteed API.\nThe value should be a CamelCase string.\nThis field may not be empty.","maxLength":1024,"minLength":1,"pattern":"^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$","type":"string"},"status":{"description":"status of the condition, one of True, False, Unknown.","enum":["True","False","Unknown"],"type":"string"},"type":{"description":"type of condition in CamelCase or in foo.example.com/CamelCase.","maxLength":316,"pattern":"^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$","type":"string"}},"required":["lastTransitionTime","message","reason","status","type"],"type":"object"},"type":"array"},"namespace":{"description":"Namespace where console plugin and flowlogs-pipeline have been deployed.\nDeprecated: annotations are used instead","type":"string"}},"required":["conditions"],"type":"object"}},"type":"object"}},"served":true,"storage":true,"subresources":{"status":{}}}]},"status":{"acceptedNames":{"kind":"","plural":""},"conditions":null,"storedVersions":null}} +- type: olm.bundle.object + value: + data: {"apiVersion":"apiextensions.k8s.io/v1","kind":"CustomResourceDefinition","metadata":{"annotations":{"controller-gen.kubebuilder.io/version":"v0.16.2"},"creationTimestamp":null,"name":"flowmetrics.flows.netobserv.io"},"spec":{"group":"flows.netobserv.io","names":{"kind":"FlowMetric","listKind":"FlowMetricList","plural":"flowmetrics","singular":"flowmetric"},"scope":"Namespaced","versions":[{"additionalPrinterColumns":[{"jsonPath":".status.prometheusName","name":"Metric Name","type":"string"},{"jsonPath":".status.conditions[?(@.type==\"Ready\")].reason","name":"Status","type":"string"},{"jsonPath":".status.conditions[?(@.type==\"CardinalityWarning\")].reason","name":"Cardinality","type":"string"}],"name":"v1alpha1","schema":{"openAPIV3Schema":{"description":"FlowMetric is the API allowing to create custom metrics from the collected flow logs.","properties":{"apiVersion":{"description":"APIVersion defines the versioned schema of this representation of an object.\nServers should convert recognized schemas to the latest internal value, and\nmay reject unrecognized values.\nMore info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources","type":"string"},"kind":{"description":"Kind is a string value representing the REST resource this object represents.\nServers may infer this from the endpoint the client submits requests to.\nCannot be updated.\nIn CamelCase.\nMore info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds","type":"string"},"metadata":{"type":"object"},"spec":{"description":"FlowMetricSpec defines the desired state of FlowMetric\nThe provided API allows you to customize these metrics according to your needs.\u003cbr\u003e\nWhen adding new metrics or modifying existing labels, you must carefully monitor the memory\nusage of Prometheus workloads as this could potentially have a high impact. Cf https://rhobs-handbook.netlify.app/products/openshiftmonitoring/telemetry.md/#what-is-the-cardinality-of-a-metric\u003cbr\u003e\nTo check the cardinality of all NetObserv metrics, run as `promql`: `count({__name__=~\"netobserv.*\"}) by (__name__)`.","properties":{"buckets":{"description":"A list of buckets to use when `type` is \"Histogram\". The list must be parsable as floats. When not set, Prometheus default buckets are used.","items":{"type":"string"},"type":"array"},"charts":{"description":"Charts configuration, for the OpenShift Console in the administrator view, Dashboards menu.","items":{"description":"Configures charts / dashboard generation associated to a metric","properties":{"dashboardName":{"default":"Main","description":"Name of the containing dashboard. If this name does not refer to an existing dashboard, a new dashboard is created.","type":"string"},"queries":{"description":"List of queries to be displayed on this chart. If `type` is `SingleStat` and multiple queries are provided,\nthis chart is automatically expanded in several panels (one per query).","items":{"description":"Configures PromQL queries","properties":{"legend":{"description":"The query legend that applies to each timeseries represented in this chart. When multiple timeseries are displayed, you should set a legend\nthat distinguishes each of them. It can be done with the following format: `{{ Label }}`. For example, if the `promQL` groups timeseries per\nlabel such as: `sum(rate($METRIC[2m])) by (Label1, Label2)`, you may write as the legend: `Label1={{ Label1 }}, Label2={{ Label2 }}`.","type":"string"},"promQL":{"description":"The `promQL` query to be run against Prometheus. If the chart `type` is `SingleStat`, this query should only return\na single timeseries. For other types, a top 7 is displayed.\nYou can use `$METRIC` to refer to the metric defined in this resource. For example: `sum(rate($METRIC[2m]))`.\nTo learn more about `promQL`, refer to the Prometheus documentation: https://prometheus.io/docs/prometheus/latest/querying/basics/","type":"string"},"top":{"default":7,"description":"Top N series to display per timestamp. Does not apply to `SingleStat` chart type.","minimum":1,"type":"integer"}},"required":["legend","promQL","top"],"type":"object"},"type":"array"},"sectionName":{"description":"Name of the containing dashboard section. If this name does not refer to an existing section, a new section is created.\nIf `sectionName` is omitted or empty, the chart is placed in the global top section.","type":"string"},"title":{"description":"Title of the chart.","type":"string"},"type":{"description":"Type of the chart.","enum":["SingleStat","Line","StackArea"],"type":"string"},"unit":{"description":"Unit of this chart. Only a few units are currently supported. Leave empty to use generic number.","enum":["bytes","seconds","Bps","pps","percent",""],"type":"string"}},"required":["dashboardName","queries","title","type"],"type":"object"},"type":"array"},"direction":{"default":"Any","description":"Filter for ingress, egress or any direction flows.\nWhen set to `Ingress`, it is equivalent to adding the regular expression filter on `FlowDirection`: `0|2`.\nWhen set to `Egress`, it is equivalent to adding the regular expression filter on `FlowDirection`: `1|2`.","enum":["Any","Egress","Ingress"],"type":"string"},"divider":{"description":"When nonzero, scale factor (divider) of the value. Metric value = Flow value / Divider.","type":"string"},"filters":{"description":"`filters` is a list of fields and values used to restrict which flows are taken into account.\nRefer to the documentation for the list of available fields: https://docs.openshift.com/container-platform/latest/observability/network_observability/json-flows-format-reference.html.","items":{"properties":{"field":{"description":"Name of the field to filter on","type":"string"},"matchType":{"default":"Equal","description":"Type of matching to apply","enum":["Equal","NotEqual","Presence","Absence","MatchRegex","NotMatchRegex"],"type":"string"},"value":{"description":"Value to filter on. When `matchType` is `Equal` or `NotEqual`, you can use field injection with `$(SomeField)` to refer to any other field of the flow.","type":"string"}},"required":["field","matchType"],"type":"object"},"type":"array"},"flatten":{"description":"`flatten` is a list of array-type fields that must be flattened, such as Interfaces or NetworkEvents. Flattened fields generate one metric per item in that field.\nFor instance, when flattening `Interfaces` on a bytes counter, a flow having Interfaces [br-ex, ens5] increases one counter for `br-ex` and another for `ens5`.","items":{"type":"string"},"type":"array"},"labels":{"description":"`labels` is a list of fields that should be used as Prometheus labels, also known as dimensions.\nFrom choosing labels results the level of granularity of this metric, and the available aggregations at query time.\nIt must be done carefully as it impacts the metric cardinality (cf https://rhobs-handbook.netlify.app/products/openshiftmonitoring/telemetry.md/#what-is-the-cardinality-of-a-metric).\nIn general, avoid setting very high cardinality labels such as IP or MAC addresses.\n\"SrcK8S_OwnerName\" or \"DstK8S_OwnerName\" should be preferred over \"SrcK8S_Name\" or \"DstK8S_Name\" as much as possible.\nRefer to the documentation for the list of available fields: https://docs.openshift.com/container-platform/latest/observability/network_observability/json-flows-format-reference.html.","items":{"type":"string"},"type":"array"},"metricName":{"description":"Name of the metric. In Prometheus, it is automatically prefixed with \"netobserv_\".","pattern":"^[a-zA-Z_][a-zA-Z0-9:_]*$|^$","type":"string"},"remap":{"additionalProperties":{"pattern":"^[a-zA-Z_][a-zA-Z0-9_]*$","type":"string"},"description":"Set the `remap` property to use different names for the generated metric labels than the flow fields. Use the origin flow fields as keys, and the desired label names as values.","type":"object"},"type":{"description":"Metric type: \"Counter\", \"Histogram\" or \"Gauge\".\nUse \"Counter\" for any value that increases over time and on which you can compute a rate, such as Bytes or Packets.\nUse \"Histogram\" for any value that must be sampled independently, such as latencies.\nUse \"Gauge\" for other values that don't necessitate accuracy over time (gauges are sampled only every N seconds when Prometheus fetches the metric).","enum":["Counter","Histogram"],"type":"string"},"valueField":{"description":"`valueField` is the flow field that must be used as a value for this metric. This field must hold numeric values.\nLeave empty to count flows rather than a specific value per flow.\nRefer to the documentation for the list of available fields: https://docs.openshift.com/container-platform/latest/observability/network_observability/json-flows-format-reference.html.","type":"string"}},"required":["type"],"type":"object"},"status":{"description":"FlowMetricStatus defines the observed state of FlowMetric","properties":{"conditions":{"description":"`conditions` represent the latest available observations of an object's state","items":{"description":"Condition contains details for one aspect of the current state of this API Resource.","properties":{"lastTransitionTime":{"description":"lastTransitionTime is the last time the condition transitioned from one status to another.\nThis should be when the underlying condition changed.  If that is not known, then using the time when the API field changed is acceptable.","format":"date-time","type":"string"},"message":{"description":"message is a human readable message indicating details about the transition.\nThis may be an empty string.","maxLength":32768,"type":"string"},"observedGeneration":{"description":"observedGeneration represents the .metadata.generation that the condition was set based upon.\nFor instance, if .metadata.generation is currently 12, but the .status.conditions[x].observedGeneration is 9, the condition is out of date\nwith respect to the current state of the instance.","format":"int64","minimum":0,"type":"integer"},"reason":{"description":"reason contains a programmatic identifier indicating the reason for the condition's last transition.\nProducers of specific condition types may define expected values and meanings for this field,\nand whether the values are considered a guaranteed API.\nThe value should be a CamelCase string.\nThis field may not be empty.","maxLength":1024,"minLength":1,"pattern":"^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$","type":"string"},"status":{"description":"status of the condition, one of True, False, Unknown.","enum":["True","False","Unknown"],"type":"string"},"type":{"description":"type of condition in CamelCase or in foo.example.com/CamelCase.","maxLength":316,"pattern":"^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$","type":"string"}},"required":["lastTransitionTime","message","reason","status","type"],"type":"object"},"type":"array"},"prometheusName":{"description":"Metric name, including prefix, as it appears in Prometheus","type":"string"}},"required":["conditions"],"type":"object"}},"type":"object"}},"served":true,"storage":true,"subresources":{"status":{}}}]},"status":{"acceptedNames":{"kind":"","plural":""},"conditions":null,"storedVersions":null}} +- type: olm.bundle.object + value: + data: {"apiVersion":"operators.coreos.com/v1","kind":"ClusterServiceVersion","metadata":{"annotations":{"alm-examples":"[\n  {\n    \"apiVersion\": \"flows.netobserv.io/v1alpha1\",\n    \"kind\": \"FlowMetric\",\n    \"metadata\": {\n      \"labels\": {\n        \"app.kubernetes.io/created-by\": \"netobserv-operator\",\n        \"app.kubernetes.io/instance\": \"flowmetric-sample\",\n        \"app.kubernetes.io/managed-by\": \"kustomize\",\n        \"app.kubernetes.io/name\": \"flowmetric\",\n        \"app.kubernetes.io/part-of\": \"netobserv-operator\"\n      },\n      \"name\": \"flowmetric-sample\"\n    },\n    \"spec\": {\n      \"charts\": [\n        {\n          \"dashboardName\": \"Main\",\n          \"queries\": [\n            {\n              \"legend\": \"\",\n              \"promQL\": \"sum(rate($METRIC[2m]))\"\n            }\n          ],\n          \"title\": \"External ingress traffic\",\n          \"type\": \"SingleStat\",\n          \"unit\": \"Bps\"\n        },\n        {\n          \"dashboardName\": \"Main\",\n          \"queries\": [\n            {\n              \"legend\": \"{{DstK8S_Namespace}} / {{DstK8S_OwnerName}}\",\n              \"promQL\": \"sum(rate($METRIC{DstK8S_Namespace!=\\\"\\\"}[2m])) by (DstK8S_Namespace, DstK8S_OwnerName)\"\n            }\n          ],\n          \"sectionName\": \"External\",\n          \"title\": \"Top external ingress traffic per workload\",\n          \"type\": \"StackArea\",\n          \"unit\": \"Bps\"\n        }\n      ],\n      \"direction\": \"Ingress\",\n      \"filters\": [\n        {\n          \"field\": \"SrcSubnetLabel\",\n          \"matchType\": \"Absence\"\n        }\n      ],\n      \"labels\": [\n        \"DstK8S_HostName\",\n        \"DstK8S_Namespace\",\n        \"DstK8S_OwnerName\",\n        \"DstK8S_OwnerType\"\n      ],\n      \"metricName\": \"cluster_external_ingress_bytes_total\",\n      \"type\": \"Counter\",\n      \"valueField\": \"Bytes\"\n    }\n  },\n  {\n    \"apiVersion\": \"flows.netobserv.io/v1beta2\",\n    \"kind\": \"FlowCollector\",\n    \"metadata\": {\n      \"name\": \"cluster\"\n    },\n    \"spec\": {\n      \"agent\": {\n        \"ebpf\": {\n          \"cacheActiveTimeout\": \"5s\",\n          \"cacheMaxFlows\": 100000,\n          \"excludeInterfaces\": [\n            \"lo\"\n          ],\n          \"imagePullPolicy\": \"IfNotPresent\",\n          \"interfaces\": [],\n          \"kafkaBatchSize\": 1048576,\n          \"logLevel\": \"info\",\n          \"metrics\": {\n            \"server\": {\n              \"port\": 9400\n            }\n          },\n          \"privileged\": false,\n          \"resources\": {\n            \"limits\": {\n              \"memory\": \"800Mi\"\n            },\n            \"requests\": {\n              \"cpu\": \"100m\",\n              \"memory\": \"50Mi\"\n            }\n          },\n          \"sampling\": 50\n        },\n        \"type\": \"eBPF\"\n      },\n      \"consolePlugin\": {\n        \"autoscaler\": {\n          \"maxReplicas\": 3,\n          \"metrics\": [\n            {\n              \"resource\": {\n                \"name\": \"cpu\",\n                \"target\": {\n                  \"averageUtilization\": 50,\n                  \"type\": \"Utilization\"\n                }\n              },\n              \"type\": \"Resource\"\n            }\n          ],\n          \"minReplicas\": 1,\n          \"status\": \"Disabled\"\n        },\n        \"enable\": true,\n        \"imagePullPolicy\": \"IfNotPresent\",\n        \"logLevel\": \"info\",\n        \"portNaming\": {\n          \"enable\": true,\n          \"portNames\": {\n            \"3100\": \"loki\"\n          }\n        },\n        \"quickFilters\": [\n          {\n            \"default\": true,\n            \"filter\": {\n              \"flow_layer\": \"\\\"app\\\"\"\n            },\n            \"name\": \"Applications\"\n          },\n          {\n            \"filter\": {\n              \"flow_layer\": \"\\\"infra\\\"\"\n            },\n            \"name\": \"Infrastructure\"\n          },\n          {\n            \"default\": true,\n            \"filter\": {\n              \"dst_kind\": \"\\\"Pod\\\"\",\n              \"src_kind\": \"\\\"Pod\\\"\"\n            },\n            \"name\": \"Pods network\"\n          },\n          {\n            \"filter\": {\n              \"dst_kind\": \"\\\"Service\\\"\"\n            },\n            \"name\": \"Services network\"\n          }\n        ],\n        \"replicas\": 1,\n        \"resources\": {\n          \"limits\": {\n            \"memory\": \"100Mi\"\n          },\n          \"requests\": {\n            \"cpu\": \"100m\",\n            \"memory\": \"50Mi\"\n          }\n        }\n      },\n      \"deploymentModel\": \"Direct\",\n      \"exporters\": [],\n      \"kafka\": {\n        \"address\": \"kafka-cluster-kafka-bootstrap.netobserv\",\n        \"tls\": {\n          \"caCert\": {\n            \"certFile\": \"ca.crt\",\n            \"name\": \"kafka-cluster-cluster-ca-cert\",\n            \"type\": \"secret\"\n          },\n          \"enable\": false,\n          \"userCert\": {\n            \"certFile\": \"user.crt\",\n            \"certKey\": \"user.key\",\n            \"name\": \"flp-kafka\",\n            \"type\": \"secret\"\n          }\n        },\n        \"topic\": \"network-flows\"\n      },\n      \"loki\": {\n        \"enable\": true,\n        \"lokiStack\": {\n          \"name\": \"loki\"\n        },\n        \"mode\": \"Monolithic\",\n        \"monolithic\": {\n          \"tenantID\": \"netobserv\",\n          \"tls\": {\n            \"caCert\": {\n              \"certFile\": \"service-ca.crt\",\n              \"name\": \"loki-gateway-ca-bundle\",\n              \"type\": \"configmap\"\n            },\n            \"enable\": false\n          },\n          \"url\": \"http://loki.netobserv.svc:3100/\"\n        },\n        \"readTimeout\": \"30s\",\n        \"writeBatchSize\": 10485760,\n        \"writeBatchWait\": \"1s\",\n        \"writeTimeout\": \"10s\"\n      },\n      \"namespace\": \"netobserv\",\n      \"networkPolicy\": {\n        \"additionalNamespaces\": [],\n        \"enable\": false\n      },\n      \"processor\": {\n        \"imagePullPolicy\": \"IfNotPresent\",\n        \"kafkaConsumerAutoscaler\": null,\n        \"kafkaConsumerBatchSize\": 10485760,\n        \"kafkaConsumerQueueCapacity\": 1000,\n        \"kafkaConsumerReplicas\": 3,\n        \"logLevel\": \"info\",\n        \"logTypes\": \"Flows\",\n        \"metrics\": {\n          \"disableAlerts\": [],\n          \"server\": {\n            \"port\": 9401\n          }\n        },\n        \"resources\": {\n          \"limits\": {\n            \"memory\": \"800Mi\"\n          },\n          \"requests\": {\n            \"cpu\": \"100m\",\n            \"memory\": \"100Mi\"\n          }\n        }\n      },\n      \"prometheus\": {\n        \"querier\": {\n          \"enable\": true,\n          \"mode\": \"Auto\",\n          \"timeout\": \"30s\"\n        }\n      }\n    }\n  }\n]","capabilities":"Seamless Upgrades","categories":"Monitoring, Networking, Observability","console.openshift.io/plugins":"[\"netobserv-plugin\"]","containerImage":"quay.io/netobserv/network-observability-operator:1.9.2-community","createdAt":"2025-09-08T08:23:26Z","description":"Network flows collector and monitoring solution","operatorframework.io/initialization-resource":"{\"apiVersion\":\"flows.netobserv.io/v1beta2\", \"kind\":\"FlowCollector\",\"metadata\":{\"name\":\"cluster\"},\"spec\": {}}","operatorframework.io/suggested-namespace":"openshift-netobserv-operator","operators.operatorframework.io/builder":"operator-sdk-v1.40.0","operators.operatorframework.io/project_layout":"go.kubebuilder.io/v4","repository":"https://github.com/netobserv/network-observability-operator","support":"NetObserv team"},"labels":{"operatorframework.io/arch.amd64":"supported","operatorframework.io/arch.arm64":"supported","operatorframework.io/arch.ppc64le":"supported","operatorframework.io/arch.s390x":"supported","operatorframework.io/os.linux":"supported"},"name":"netobserv-operator.v1.9.2-community","namespace":"placeholder"},"spec":{"apiservicedefinitions":{},"customresourcedefinitions":{"owned":[{"description":"`FlowCollector` is the schema for the network flows collection API, which pilots and configures the underlying deployments.","displayName":"Flow Collector","kind":"FlowCollector","name":"flowcollectors.flows.netobserv.io","specDescriptors":[{"description":"defines the desired type of deployment for flow processing.","displayName":"Deployment model","path":"deploymentModel"},{"description":"for flows extraction.","displayName":"Agent configuration","path":"agent"},{"path":"agent.type","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"path":"agent.ipfix","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"description":"Settings related to the eBPF-based flow reporter.","displayName":"eBPF Agent configuration","path":"agent.ebpf","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:agent.type:eBPF"]},{"displayName":"Privileged mode","path":"agent.ebpf.privileged","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:booleanSwitch"]},{"displayName":"Cache active timeout","path":"agent.ebpf.cacheActiveTimeout","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Cache max flows","path":"agent.ebpf.cacheMaxFlows","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Kafka batch size","path":"agent.ebpf.kafkaBatchSize","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Log level","path":"agent.ebpf.logLevel","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Image pull policy","path":"agent.ebpf.imagePullPolicy","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:imagePullPolicy","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Resource Requirements","path":"agent.ebpf.resources","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:resourceRequirements"]},{"path":"agent.ebpf.advanced","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"path":"agent.ebpf.flowFilter","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"path":"agent.ebpf.metrics.enable","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"description":"to use Kafka as a broker as part of the flow collection pipeline.","displayName":"Kafka configuration","path":"kafka","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka"]},{"displayName":"TLS configuration","path":"kafka.tls"},{"path":"kafka.tls.enable","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:booleanSwitch"]},{"displayName":"Insecure","path":"kafka.tls.insecureSkipVerify","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:booleanSwitch","urn:alm:descriptor:com.tectonic.ui:fieldDependency:kafka.tls.enable:true"]},{"displayName":"User certificate when using mTLS","path":"kafka.tls.userCert","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:kafka.tls.enable:true"]},{"displayName":"CA certificate","path":"kafka.tls.caCert","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:kafka.tls.enable:true"]},{"path":"kafka.sasl","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"description":"of the component that receives the flows from the agent, enriches them, generates metrics, and forwards them to the Loki persistence layer and/or any available exporter.","displayName":"Processor configuration","path":"processor"},{"displayName":"Multi-cluster deployment","path":"processor.multiClusterDeployment","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced","urn:alm:descriptor:com.tectonic.ui:booleanSwitch"]},{"displayName":"Cluster name","path":"processor.clusterName","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced","urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.multiClusterDeployment:true"]},{"displayName":"Availability zones","path":"processor.addZone","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:booleanSwitch"]},{"path":"processor.advanced","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"displayName":"Metrics configuration","path":"processor.metrics"},{"displayName":"Server configuration","path":"processor.metrics.server"},{"displayName":"TLS configuration","path":"processor.metrics.server.tls"},{"displayName":"Insecure","path":"processor.metrics.server.tls.insecureSkipVerify","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.metrics.server.tls.type:Provided"]},{"displayName":"Cert","path":"processor.metrics.server.tls.provided","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.metrics.server.tls.type:Provided"]},{"displayName":"CA","path":"processor.metrics.server.tls.providedCaFile","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.metrics.server.tls.type:Provided"]},{"displayName":"Kafka consumer replicas","path":"processor.kafkaConsumerReplicas","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"kafka consumer autoscaler","path":"processor.kafkaConsumerAutoscaler","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Kafka consumer queue capacity","path":"processor.kafkaConsumerQueueCapacity","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Kafka consumer batch size","path":"processor.kafkaConsumerBatchSize","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"path":"processor.subnetLabels.openShiftAutoDetect","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"displayName":"Log level","path":"processor.logLevel","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Image pull policy","path":"processor.imagePullPolicy","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:imagePullPolicy","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Resource Requirements","path":"processor.resources","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:resourceRequirements"]},{"description":"for the flow store.","displayName":"Loki client settings","path":"loki"},{"displayName":"Enable","path":"loki.enable","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:booleanSwitch"]},{"displayName":"Mode","path":"loki.mode","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true"]},{"displayName":"Loki stack","path":"loki.lokiStack","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:LokiStack"]},{"displayName":"Monolithic","path":"loki.monolithic","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:Monolithic"]},{"displayName":"Microservices","path":"loki.microservices","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:Microservices"]},{"displayName":"Manual","path":"loki.manual","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:Manual"]},{"displayName":"Write batch wait","path":"loki.writeBatchWait","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Write batch size","path":"loki.writeBatchSize","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Write timeout","path":"loki.writeTimeout","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"path":"loki.advanced","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"description":"related to the OpenShift Console integration.","displayName":"Console plugin configuration","path":"consolePlugin","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true"]},{"displayName":"Enable","path":"consolePlugin.enable","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:booleanSwitch"]},{"displayName":"Port naming","path":"consolePlugin.portNaming","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true"]},{"displayName":"Quick filters","path":"consolePlugin.quickFilters","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true"]},{"displayName":"Replicas","path":"consolePlugin.replicas","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Horizontal pod autoscaler","path":"consolePlugin.autoscaler","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Log level","path":"consolePlugin.logLevel","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Image pull policy","path":"consolePlugin.imagePullPolicy","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:imagePullPolicy\"","urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true","urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Resource Requirements","path":"consolePlugin.resources","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:resourceRequirements","urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true"]},{"path":"consolePlugin.advanced","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:hidden"]},{"description":"additional optional exporters for custom consumption or storage.","displayName":"Exporters","path":"exporters","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:advanced"]},{"displayName":"Type","path":"exporters[0].type"},{"displayName":"IPFIX configuration","path":"exporters[0].ipfix","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:exporters.type:IPFIX"]},{"displayName":"Kafka configuration","path":"exporters[0].kafka","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:exporters.type:Kafka"]},{"displayName":"OpenTelemetry configuration","path":"exporters[0].openTelemetry","x-descriptors":["urn:alm:descriptor:com.tectonic.ui:fieldDependency:exporters.type:OpenTelemetry"]},{"displayName":"Exclude interfaces","path":"agent.ebpf.excludeInterfaces"},{"displayName":"Features","path":"agent.ebpf.features"},{"displayName":"Interfaces","path":"agent.ebpf.interfaces"},{"displayName":"Metrics","path":"agent.ebpf.metrics"},{"displayName":"Disable alerts","path":"agent.ebpf.metrics.disableAlerts"},{"displayName":"Server","path":"agent.ebpf.metrics.server"},{"displayName":"Port","path":"agent.ebpf.metrics.server.port"},{"displayName":"Sampling","path":"agent.ebpf.sampling"},{"displayName":"Enable","path":"consolePlugin.portNaming.enable"},{"displayName":"Port names","path":"consolePlugin.portNaming.portNames"},{"displayName":"Address","path":"kafka.address"},{"displayName":"Topic","path":"kafka.topic"},{"displayName":"Name","path":"loki.lokiStack.name"},{"displayName":"Namespace","path":"loki.lokiStack.namespace"},{"displayName":"Auth token","path":"loki.manual.authToken"},{"displayName":"Ingester url","path":"loki.manual.ingesterUrl"},{"displayName":"Querier url","path":"loki.manual.querierUrl"},{"displayName":"Status url","path":"loki.manual.statusUrl"},{"displayName":"Tenant id","path":"loki.manual.tenantID"},{"displayName":"Ingester url","path":"loki.microservices.ingesterUrl"},{"displayName":"Querier url","path":"loki.microservices.querierUrl"},{"displayName":"Tenant id","path":"loki.microservices.tenantID"},{"displayName":"Tenant id","path":"loki.monolithic.tenantID"},{"displayName":"Url","path":"loki.monolithic.url"},{"displayName":"Read timeout","path":"loki.readTimeout"},{"displayName":"Namespace","path":"namespace"},{"displayName":"Network policy","path":"networkPolicy"},{"displayName":"Additional namespaces","path":"networkPolicy.additionalNamespaces"},{"displayName":"Enable","path":"networkPolicy.enable"},{"displayName":"Deduper","path":"processor.deduper"},{"displayName":"Mode","path":"processor.deduper.mode"},{"displayName":"Sampling","path":"processor.deduper.sampling"},{"displayName":"Filters","path":"processor.filters"},{"displayName":"Log types","path":"processor.logTypes"},{"displayName":"Disable alerts","path":"processor.metrics.disableAlerts"},{"displayName":"Include list","path":"processor.metrics.includeList"},{"displayName":"Port","path":"processor.metrics.server.port"},{"displayName":"Subnet labels","path":"processor.subnetLabels"},{"displayName":"Custom labels","path":"processor.subnetLabels.customLabels"},{"displayName":"Prometheus","path":"prometheus"},{"displayName":"Querier","path":"prometheus.querier"},{"displayName":"Enable","path":"prometheus.querier.enable"},{"displayName":"Manual","path":"prometheus.querier.manual"},{"displayName":"Forward user token","path":"prometheus.querier.manual.forwardUserToken"},{"displayName":"Url","path":"prometheus.querier.manual.url"},{"displayName":"Mode","path":"prometheus.querier.mode"},{"displayName":"Timeout","path":"prometheus.querier.timeout"}],"statusDescriptors":[{"description":"Namespace where console plugin and flowlogs-pipeline have been deployed.","displayName":"Namespace","path":"namespace","x-descriptors":["urn:alm:descriptor:text"]},{"description":"Conditions of the FlowCollector instance health.","displayName":"Conditions","path":"conditions","x-descriptors":["urn:alm:descriptor:io.kubernetes.conditions"]}],"version":"v1beta2"},{"description":"`FlowMetric` is the schema for the custom metrics API, which allows to generate more metrics out of flow logs. You can find examples here: https://github.com/netobserv/network-observability-operator/tree/main/config/samples/flowmetrics","displayName":"Flow Metric","kind":"FlowMetric","name":"flowmetrics.flows.netobserv.io","version":"v1alpha1"}]},"description":"NetObserv Operator is an OpenShift / Kubernetes operator for network observability. It deploys a monitoring pipeline that consists in:\n- an eBPF agent, that generates network flows from captured packets\n- flowlogs-pipeline, a component that collects, enriches and exports these flows\n- when used in OpenShift, a Console plugin for flows visualization with powerful filtering options, a topology representation and more\n\nFlow data is then available in multiple ways, each optional:\n\n- As Prometheus metrics\n- As raw flow logs stored in Grafana Loki\n- As raw flow logs exported to a collector\n\n## Dependencies\n\n### Loki\n\n[Loki](https://grafana.com/oss/loki/), from GrafanaLabs, can optionally be used as the backend to store all collected flows. The NetObserv Operator does not install Loki directly, however we provide some guidance to help you there.\n\nFor normal usage, we recommend two options:\n\n- Installing the [Loki Operator](https://loki-operator.dev/docs/prologue/quickstart.md/). We have written [a guide](https://github.com/netobserv/documents/blob/main/loki_operator.md) to help you through those steps. Please note that it requires configuring an object storage. Note also that the Loki Operator can also be used for [OpenShift cluster logging](https://docs.openshift.com/container-platform/latest/logging/cluster-logging.html). If you do so, you should not share the same `LokiStack` for Logging and NetObserv.\n\n- Installing using [Grafana's official documentation](https://grafana.com/docs/loki/latest/). Here also we wrote a [\"distributed Loki\" step by step guide](https://github.com/netobserv/documents/blob/main/loki_distributed.md).\n\nFor a quick try that is not suitable for production and not scalable (it deploys a single pod, configures a 10GB storage PVC, with 24 hours of retention), you can simply run the following commands:\n\n```\nkubectl create namespace netobserv\nkubectl apply -f \u003c(curl -L https://raw.githubusercontent.com/netobserv/documents/5410e65b8e05aaabd1244a9524cfedd8ac8c56b5/examples/zero-click-loki/1-storage.yaml) -n netobserv\nkubectl apply -f \u003c(curl -L https://raw.githubusercontent.com/netobserv/documents/5410e65b8e05aaabd1244a9524cfedd8ac8c56b5/examples/zero-click-loki/2-loki.yaml) -n netobserv\n```\n\nIf you prefer to not use Loki, you must set `spec.loki.enable` to `false` in `FlowCollector`.\nIn that case, you can still get the Prometheus metrics or export raw flows to a custom collector. But be aware that some of the Console plugin features will be disabled. For instance, you will not be able to view raw flows there, and the metrics / topology will have a more limited level of details, missing information such as pods or IPs.\n\n### Kafka\n\n[Apache Kafka](https://kafka.apache.org/) can optionally be used for a more resilient and scalable architecture. You can use for example [Strimzi](https://strimzi.io/), an operator-based distribution of Kafka for Kubernetes and OpenShift.\n\n### Grafana\n\n[Grafana](https://grafana.com/oss/grafana/) can optionally be installed for custom dashboards and query capabilities.\n\n## Configuration\n\nThe `FlowCollector` resource is used to configure the operator and its managed components. A comprehensive documentation is [available here](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/docs/FlowCollector.md), and a full sample file [there](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/config/samples/flows_v1beta2_flowcollector.yaml).\n\nTo edit configuration in cluster, run:\n\n```bash\nkubectl edit flowcollector cluster\n```\n\nAs it operates cluster-wide on every node, only a single `FlowCollector` is allowed, and it has to be named `cluster`.\n\nA couple of settings deserve special attention:\n\n- Sampling (`spec.agent.ebpf.sampling`): a value of `100` means: one flow every 100 is sampled. `1` means all flows are sampled. The lower it is, the more flows you get, and the more accurate are derived metrics, but the higher amount of resources are consumed. By default, sampling is set to 50 (ie. 1:50). Note that more sampled flows also means more storage needed. We recommend to start with default values and refine empirically, to figure out which setting your cluster can manage.\n\n- Loki (`spec.loki`): configure here how to reach Loki. The default values match the Loki quick install paths mentioned above, but you might have to configure differently if you used another installation method. Make sure to disable it (`spec.loki.enable`) if you don't want to use Loki.\n\n- Quick filters (`spec.consolePlugin.quickFilters`): configure preset filters to be displayed in the Console plugin. They offer a way to quickly switch from filters to others, such as showing / hiding pods network, or infrastructure network, or application network, etc. They can be tuned to reflect the different workloads running on your cluster. For a list of available filters, [check this page](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/docs/QuickFilters.md).\n\n- Kafka (`spec.deploymentModel: KAFKA` and `spec.kafka`): when enabled, integrates the flow collection pipeline with Kafka, by splitting ingestion from transformation (kube enrichment, derived metrics, ...). Kafka can provide better scalability, resiliency and high availability ([view more details](https://www.redhat.com/en/topics/integration/what-is-apache-kafka)). Assumes Kafka is already deployed and a topic is created.\n\n- Exporters (`spec.exporters`) an optional list of exporters to which to send enriched flows. KAFKA and IPFIX exporters are supported. This allows you to define any custom storage or processing that can read from Kafka or use the IPFIX standard.\n\n- To enable availability zones awareness, set `spec.processor.addZone` to `true`.\n\n## Resource considerations\n\nThe following table outlines examples of resource considerations for clusters with certain workload sizes.\nThe examples outlined in the table demonstrate scenarios that are tailored to specific workloads. Consider each example only as a baseline from which adjustments can be made to accommodate your workload needs.\n\n\n| Resource recommendations                        | Extra small (10 nodes) | Small (25 nodes)       | Medium (65 nodes) **    | Large (120 nodes) **          |\n| ----------------------------------------------- | ---------------------- | ---------------------- | ----------------------- | ----------------------------- |\n| *Worker Node vCPU and memory*                   | 4 vCPUs\\| 16GiB mem *  | 16 vCPUs\\| 64GiB mem * | 16 vCPUs\\| 64GiB mem  * |16 vCPUs\\| 64GiB Mem *         |\n| *LokiStack size*                                | `1x.extra-small`       | `1x.small`             | `1x.small`              | `1x.medium`                   |\n| *Network Observability controller memory limit* | 400Mi (default)        | 400Mi (default)        | 400Mi (default)         | 800Mi                         |\n| *eBPF sampling rate*                            | 50 (default)           | 50 (default)           | 50 (default)            | 50 (default)                  |\n| *eBPF memory limit*                             | 800Mi (default)        | 800Mi (default)        | 2000Mi                  | 800Mi (default)               |\n| *FLP memory limit*                              | 800Mi (default)        | 800Mi (default)        | 800Mi (default)         | 800Mi (default)               |\n| *FLP Kafka partitions*                          | N/A                    | 48                     | 48                      | 48                            |\n| *Kafka consumer replicas*                       | N/A                    | 24                     | 24                      | 24                            |\n| *Kafka brokers*                                 | N/A                    | 3 (default)            | 3 (default)             | 3 (default)                   |\n\n*. Tested with AWS M6i instances.\n**. In addition to this worker and its controller, 3 infra nodes (size `M6i.12xlarge`) and 1 workload node (size `M6i.8xlarge`) were tested.\n\n## Further reading\n\nPlease refer to the documentation on GitHub for more information.\n\nThis documentation includes:\n\n- An [overview](https://github.com/netobserv/network-observability-operator#openshift-console) of the features, with screenshots\n- More information on [configuring metrics](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/docs/Metrics.md).\n- A [performance](https://github.com/netobserv/network-observability-operator#performance-fine-tuning) section, for fine-tuning\n- A [security](https://github.com/netobserv/network-observability-operator#securing-data-and-communications) section\n- An [F.A.Q.](https://github.com/netobserv/network-observability-operator#faq--troubleshooting) section","displayName":"NetObserv Operator","icon":[{"base64data":"PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPCEtLSBHZW5lcmF0b3I6IEFkb2JlIElsbHVzdHJhdG9yIDI2LjAuMiwgU1ZHIEV4cG9ydCBQbHVnLUluIC4gU1ZHIFZlcnNpb246IDYuMDAgQnVpbGQgMCkgIC0tPgo8c3ZnIHZlcnNpb249IjEuMSIgaWQ9IkxheWVyXzEiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHg9IjBweCIgeT0iMHB4IgoJIHZpZXdCb3g9IjAgMCAxMDAgMTAwIiBzdHlsZT0iZW5hYmxlLWJhY2tncm91bmQ6bmV3IDAgMCAxMDAgMTAwOyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+CjxzdHlsZSB0eXBlPSJ0ZXh0L2NzcyI+Cgkuc3Qwe2ZpbGw6dXJsKCNTVkdJRF8xXyk7fQoJLnN0MXtmaWxsOiNGRkZGRkY7fQoJLnN0MntvcGFjaXR5OjAuNjt9Cgkuc3Qze29wYWNpdHk6MC41O30KCS5zdDR7b3BhY2l0eTowLjQ7fQo8L3N0eWxlPgo8Zz4KCTxnPgoJCTxnPgoJCQk8cmFkaWFsR3JhZGllbnQgaWQ9IlNWR0lEXzFfIiBjeD0iMTQuNzc1OCIgY3k9Ii0yLjk3NzEiIHI9IjkxLjYyNyIgZ3JhZGllbnRVbml0cz0idXNlclNwYWNlT25Vc2UiPgoJCQkJPHN0b3AgIG9mZnNldD0iMCIgc3R5bGU9InN0b3AtY29sb3I6IzNDM0ZBNiIvPgoJCQkJPHN0b3AgIG9mZnNldD0iMSIgc3R5bGU9InN0b3AtY29sb3I6IzNCMDM0MCIvPgoJCQk8L3JhZGlhbEdyYWRpZW50PgoJCQk8cGF0aCBjbGFzcz0ic3QwIiBkPSJNNTAsOTljLTEzLjMsMC0yNS40LTUuMy0zNC4yLTEzLjlDNi43LDc2LjIsMSw2My43LDEsNTBDMSwyMi45LDIyLjksMSw1MCwxYzEzLjcsMCwyNi4yLDUuNywzNS4xLDE0LjgKCQkJCUM5My43LDI0LjYsOTksMzYuNyw5OSw1MEM5OSw3Ny4xLDc3LjEsOTksNTAsOTl6Ii8+CgkJPC9nPgoJCTxnPgoJCQk8Y2lyY2xlIGNsYXNzPSJzdDEiIGN4PSIzNy41IiBjeT0iODEuOSIgcj0iNSIvPgoJCTwvZz4KCQk8cGF0aCBjbGFzcz0ic3QxIiBkPSJNNDguNiw5MS45bDE4LjgtNDMuM2MtMi41LTAuMS01LTAuNy03LjItMkwzMy4yLDY4LjJsMS40LTEuOGwyMC0yNS4xYy0xLjUtMi40LTIuMy01LjEtMi4zLTcuOUw5LDUyLjIKCQkJbDQ3LjYtMjkuOWwwLDBjMC4xLTAuMSwwLjItMC4yLDAuMi0wLjJjNi4xLTYuMSwxNS45LTYuMSwyMiwwbDAuMSwwLjFjNiw2LjEsNiwxNS45LTAuMSwyMS45Yy0wLjEsMC4xLTAuMiwwLjItMC4yLDAuMmwwLDAKCQkJTDQ4LjYsOTEuOXoiLz4KCQk8ZyBjbGFzcz0ic3QyIj4KCQkJPGNpcmNsZSBjbGFzcz0ic3QxIiBjeD0iNTAuMyIgY3k9IjE0LjciIHI9IjMuMSIvPgoJCTwvZz4KCQk8ZyBjbGFzcz0ic3QzIj4KCQkJPGNpcmNsZSBjbGFzcz0ic3QxIiBjeD0iMjcuNyIgY3k9IjU4IiByPSIxLjciLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9Ijc3LjQiIGN5PSI2OS4zIiByPSIxLjciLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjE2LjMiIGN5PSIzNi42IiByPSIxLjciLz4KCQk8L2c+CgkJPGcgY2xhc3M9InN0NCI+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjYzLjciIGN5PSI4NS45IiByPSIyLjIiLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjI5LjQiIGN5PSIxOS42IiByPSI0LjgiLz4KCQk8L2c+CgkJPGcgY2xhc3M9InN0MyI+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9Ijg4IiBjeT0iNTAiIHI9IjQuOCIvPgoJCTwvZz4KCTwvZz4KPC9nPgo8L3N2Zz4K","mediatype":"image/svg+xml"}],"install":{"spec":{"clusterPermissions":[{"rules":[{"apiGroups":[""],"resources":["configmaps","namespaces","secrets","serviceaccounts","services"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":[""],"resources":["endpoints","nodes","pods"],"verbs":["get","list","watch"]},{"apiGroups":["apiextensions.k8s.io"],"resources":["customresourcedefinitions"],"verbs":["get","list","watch"]},{"apiGroups":["apiextensions.k8s.io"],"resources":["customresourcedefinitions/status"],"verbs":["patch","update"]},{"apiGroups":["apiregistration.k8s.io"],"resources":["apiservices"],"verbs":["get","list","watch"]},{"apiGroups":["apps"],"resources":["daemonsets","deployments"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["apps"],"resources":["replicasets"],"verbs":["get","list","watch"]},{"apiGroups":["autoscaling"],"resources":["horizontalpodautoscalers"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["bpfman.io"],"resources":["clusterbpfapplications"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["bpfman.io"],"resources":["clusterbpfapplications/status"],"verbs":["get","patch","update"]},{"apiGroups":["config.openshift.io"],"resources":["clusterversions","networks"],"verbs":["get","list","watch"]},{"apiGroups":["console.openshift.io"],"resources":["consoleplugins"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["flows.netobserv.io"],"resources":["flowcollectors","flowmetrics"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["flows.netobserv.io"],"resources":["flowcollectors/finalizers"],"verbs":["update"]},{"apiGroups":["flows.netobserv.io"],"resources":["flowcollectors/status","flowmetrics/status"],"verbs":["get","patch","update"]},{"apiGroups":["k8s.ovn.org"],"resources":["clusteruserdefinednetworks","userdefinednetworks"],"verbs":["get","list","watch"]},{"apiGroups":["loki.grafana.com"],"resources":["lokistacks"],"verbs":["get","list","watch"]},{"apiGroups":["loki.grafana.com"],"resourceNames":["logs"],"resources":["network"],"verbs":["create"]},{"apiGroups":["metrics.k8s.io"],"resources":["pods"],"verbs":["create"]},{"apiGroups":["monitoring.coreos.com"],"resources":["prometheusrules","servicemonitors"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["networking.k8s.io"],"resources":["networkpolicies"],"verbs":["create","delete","get","list","patch","update","watch"]},{"apiGroups":["operator.openshift.io"],"resources":["consoles"],"verbs":["get","list","update","watch"]},{"apiGroups":["rbac.authorization.k8s.io"],"resources":["clusterrolebindings","rolebindings"],"verbs":["create","delete","get","list","update","watch"]},{"apiGroups":["security.openshift.io"],"resources":["securitycontextconstraints"],"verbs":["create","list","update","watch"]},{"apiGroups":["security.openshift.io"],"resourceNames":["hostnetwork"],"resources":["securitycontextconstraints"],"verbs":["use"]},{"apiGroups":["authentication.k8s.io"],"resources":["tokenreviews"],"verbs":["create"]},{"apiGroups":["authorization.k8s.io"],"resources":["subjectaccessreviews"],"verbs":["create"]}],"serviceAccountName":"netobserv-controller-manager"}],"deployments":[{"label":{"app":"netobserv-operator","control-plane":"controller-manager"},"name":"netobserv-controller-manager","spec":{"replicas":1,"selector":{"matchLabels":{"app":"netobserv-operator","control-plane":"controller-manager"}},"strategy":{},"template":{"metadata":{"labels":{"app":"netobserv-operator","control-plane":"controller-manager"}},"spec":{"containers":[{"args":["--health-probe-bind-address=:8081","--metrics-bind-address=:8443","--leader-elect","--ebpf-agent-image=$(RELATED_IMAGE_EBPF_AGENT)","--flowlogs-pipeline-image=$(RELATED_IMAGE_FLOWLOGS_PIPELINE)","--console-plugin-image=$(RELATED_IMAGE_CONSOLE_PLUGIN)","--console-plugin-compat-image=$(RELATED_IMAGE_CONSOLE_PLUGIN_COMPAT)","--namespace=$(NAMESPACE)","--downstream-deployment=$(DOWNSTREAM_DEPLOYMENT)","--profiling-bind-address=$(PROFILING_BIND_ADDRESS)","--metrics-cert-file=/etc/tls/private/tls.crt","--metrics-cert-key-file=/etc/tls/private/tls.key"],"command":["/manager"],"env":[{"name":"RELATED_IMAGE_EBPF_AGENT","value":"quay.io/netobserv/netobserv-ebpf-agent:v1.9.2-community"},{"name":"RELATED_IMAGE_FLOWLOGS_PIPELINE","value":"quay.io/netobserv/flowlogs-pipeline:v1.9.2-community"},{"name":"RELATED_IMAGE_CONSOLE_PLUGIN","value":"quay.io/netobserv/network-observability-console-plugin:v1.9.2-community"},{"name":"RELATED_IMAGE_CONSOLE_PLUGIN_COMPAT","value":"quay.io/netobserv/network-observability-console-plugin-pf4:v1.8.2-community"},{"name":"DOWNSTREAM_DEPLOYMENT","value":"false"},{"name":"PROFILING_BIND_ADDRESS"},{"name":"NAMESPACE","valueFrom":{"fieldRef":{"fieldPath":"metadata.namespace"}}}],"image":"quay.io/netobserv/network-observability-operator:1.9.2-community","imagePullPolicy":"Always","livenessProbe":{"httpGet":{"path":"/healthz","port":8081},"initialDelaySeconds":15,"periodSeconds":20},"name":"manager","ports":[{"containerPort":9443,"name":"webhook-server","protocol":"TCP"}],"readinessProbe":{"httpGet":{"path":"/readyz","port":8081},"initialDelaySeconds":5,"periodSeconds":10},"resources":{"limits":{"memory":"400Mi"},"requests":{"cpu":"100m","memory":"100Mi"}},"securityContext":{"allowPrivilegeEscalation":false,"capabilities":{"drop":["ALL"]},"readOnlyRootFilesystem":true},"volumeMounts":[{"mountPath":"/tmp/k8s-webhook-server/serving-certs","name":"cert","readOnly":true},{"mountPath":"/etc/tls/private","name":"manager-metric-tls","readOnly":true}]}],"securityContext":{"runAsNonRoot":true},"serviceAccountName":"netobserv-controller-manager","terminationGracePeriodSeconds":10,"volumes":[{"name":"cert","secret":{"defaultMode":420,"secretName":"webhook-server-cert"}},{"name":"manager-metric-tls","secret":{"defaultMode":420,"secretName":"manager-metrics-tls"}}]}}}}],"permissions":[{"rules":[{"apiGroups":[""],"resources":["configmaps"],"verbs":["get","list","watch","create","update","patch","delete"]},{"apiGroups":["coordination.k8s.io"],"resources":["leases"],"verbs":["get","list","watch","create","update","patch","delete"]},{"apiGroups":[""],"resources":["events"],"verbs":["create","patch"]}],"serviceAccountName":"netobserv-controller-manager"}]},"strategy":"deployment"},"installModes":[{"supported":false,"type":"OwnNamespace"},{"supported":false,"type":"SingleNamespace"},{"supported":false,"type":"MultiNamespace"},{"supported":true,"type":"AllNamespaces"}],"keywords":["network observability","ebpf","ipfix","flow tracing","flows","topology","network","observability"],"links":[{"name":"Project page","url":"https://github.com/netobserv/network-observability-operator"},{"name":"Issue tracker","url":"https://github.com/netobserv/network-observability-operator/issues"},{"name":"Discussion board","url":"https://github.com/netobserv/network-observability-operator/discussions"}],"maintainers":[{"email":"jpinsonn@redhat.com","name":"Julien Pinsonneau"},{"email":"jtakvori@redhat.com","name":"Joel Takvorian"},{"email":"kmeth@redhat.com","name":"Kalman Meth"},{"email":"mmahmoud@redhat.com","name":"Mohamed S. Mahmoud"},{"email":"ocazade@redhat.com","name":"Olivier Cazade"},{"email":"rschaffe@redhat.com","name":"Ronen Schaffer"},{"email":"stlee@redhat.com","name":"Steven Lee"}],"maturity":"alpha","minKubeVersion":"1.23.0","provider":{"name":"Red Hat","url":"https://www.redhat.com"},"relatedImages":[{"image":"quay.io/netobserv/netobserv-ebpf-agent:v1.9.2-community","name":"ebpf-agent"},{"image":"quay.io/netobserv/flowlogs-pipeline:v1.9.2-community","name":"flowlogs-pipeline"},{"image":"quay.io/netobserv/network-observability-console-plugin:v1.9.2-community","name":"console-plugin"},{"image":"quay.io/netobserv/network-observability-console-plugin-pf4:v1.8.2-community","name":"console-plugin-compat"}],"version":"1.9.2-community","webhookdefinitions":[{"admissionReviewVersions":["v1"],"containerPort":443,"deploymentName":"netobserv-controller-manager","failurePolicy":"Fail","generateName":"flowcollectorconversionwebhook.netobserv.io","rules":[{"apiGroups":["flows.netobserv.io"],"apiVersions":["v1beta2"],"operations":["CREATE","UPDATE"],"resources":["flowcollectors"]}],"sideEffects":"None","targetPort":9443,"type":"ValidatingAdmissionWebhook","webhookPath":"/validate-flows-netobserv-io-v1beta2-flowcollector"},{"admissionReviewVersions":["v1"],"containerPort":443,"deploymentName":"netobserv-controller-manager","failurePolicy":"Fail","generateName":"flowmetricvalidationwebhook.netobserv.io","rules":[{"apiGroups":["flows.netobserv.io"],"apiVersions":["v1alpha1"],"operations":["CREATE","UPDATE"],"resources":["flowmetrics"]}],"sideEffects":"None","targetPort":9443,"type":"ValidatingAdmissionWebhook","webhookPath":"/validate-flows-netobserv-io-v1alpha1-flowmetric"}]}} +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoibW9uaXRvcmluZy5jb3Jlb3MuY29tL3YxIiwia2luZCI6IlNlcnZpY2VNb25pdG9yIiwibWV0YWRhdGEiOnsibGFiZWxzIjp7ImNvbnRyb2wtcGxhbmUiOiJjb250cm9sbGVyLW1hbmFnZXIifSwibmFtZSI6Im5ldG9ic2Vydi1tZXRyaWNzLW1vbml0b3IifSwic3BlYyI6eyJlbmRwb2ludHMiOlt7ImJlYXJlclRva2VuRmlsZSI6Ii92YXIvcnVuL3NlY3JldHMva3ViZXJuZXRlcy5pby9zZXJ2aWNlYWNjb3VudC90b2tlbiIsImludGVydmFsIjoiMzBzIiwicGF0aCI6Ii9tZXRyaWNzIiwicG9ydCI6Imh0dHBzIiwic2NoZW1lIjoiaHR0cHMiLCJ0bHNDb25maWciOnsiY2FGaWxlIjoiL2V0Yy9wcm9tZXRoZXVzL2NvbmZpZ21hcHMvc2VydmluZy1jZXJ0cy1jYS1idW5kbGUvc2VydmljZS1jYS5jcnQiLCJzZXJ2ZXJOYW1lIjoibmV0b2JzZXJ2LW1ldHJpY3Mtc2VydmljZS5vcGVuc2hpZnQtbmV0b2JzZXJ2LW9wZXJhdG9yLnN2YyJ9fV0sInNlbGVjdG9yIjp7Im1hdGNoTGFiZWxzIjp7ImNvbnRyb2wtcGxhbmUiOiJjb250cm9sbGVyLW1hbmFnZXIifX19fQ== +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LW1ldHJpY3MtcmVhZGVyIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsibWV0cmljcy5rOHMuaW8iXSwicmVzb3VyY2VzIjpbInBvZHMiXSwidmVyYnMiOlsiY3JlYXRlIl19XX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LWNvbmZpZy13YXRjaGVyIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsiIl0sInJlc291cmNlcyI6WyJjb25maWdtYXBzIl0sInZlcmJzIjpbImdldCIsIndhdGNoIl19XX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LWV4cG9zZS1tZXRyaWNzIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsiIl0sInJlc291cmNlcyI6WyJzZXJ2aWNlcyIsImVuZHBvaW50cyIsInBvZHMiXSwidmVyYnMiOlsiZ2V0IiwibGlzdCIsIndhdGNoIl19XX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LWhvc3RuZXR3b3JrIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsic2VjdXJpdHkub3BlbnNoaWZ0LmlvIl0sInJlc291cmNlTmFtZXMiOlsiaG9zdG5ldHdvcmsiXSwicmVzb3VyY2VzIjpbInNlY3VyaXR5Y29udGV4dGNvbnN0cmFpbnRzIl0sInZlcmJzIjpbInVzZSJdfV19 +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LWluZm9ybWVycyJ9LCJydWxlcyI6W3siYXBpR3JvdXBzIjpbIiJdLCJyZXNvdXJjZXMiOlsicG9kcyIsIm5vZGVzIiwic2VydmljZXMiXSwidmVyYnMiOlsiZ2V0IiwibGlzdCIsIndhdGNoIl19LHsiYXBpR3JvdXBzIjpbImFwcHMiXSwicmVzb3VyY2VzIjpbInJlcGxpY2FzZXRzIl0sInZlcmJzIjpbImdldCIsImxpc3QiLCJ3YXRjaCJdfSx7ImFwaUdyb3VwcyI6WyJrOHMub3ZuLm9yZyJdLCJyZXNvdXJjZXMiOlsiY2x1c3RlcnVzZXJkZWZpbmVkbmV0d29ya3MiLCJ1c2VyZGVmaW5lZG5ldHdvcmtzIl0sInZlcmJzIjpbImdldCIsImxpc3QiLCJ3YXRjaCJdfV19 +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LWxva2ktcmVhZGVyIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsibG9raS5ncmFmYW5hLmNvbSJdLCJyZXNvdXJjZU5hbWVzIjpbImxvZ3MiXSwicmVzb3VyY2VzIjpbIm5ldHdvcmsiXSwidmVyYnMiOlsiZ2V0Il19XX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LWxva2ktd3JpdGVyIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsibG9raS5ncmFmYW5hLmNvbSJdLCJyZXNvdXJjZU5hbWVzIjpbImxvZ3MiXSwicmVzb3VyY2VzIjpbIm5ldHdvcmsiXSwidmVyYnMiOlsiY3JlYXRlIl19XX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJDbHVzdGVyUm9sZSIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LXRva2VuLXJldmlldyJ9LCJydWxlcyI6W3siYXBpR3JvdXBzIjpbImF1dGhlbnRpY2F0aW9uLms4cy5pbyJdLCJyZXNvdXJjZXMiOlsidG9rZW5yZXZpZXdzIl0sInZlcmJzIjpbImNyZWF0ZSJdfV19 +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJSb2xlIiwibWV0YWRhdGEiOnsiY3JlYXRpb25UaW1lc3RhbXAiOm51bGwsIm5hbWUiOiJuZXRvYnNlcnYtb3BlbnNoaWZ0LW5ldG9ic2Vydi1vcGVyYXRvci1wcm9tZXRoZXVzIn0sInJ1bGVzIjpbeyJhcGlHcm91cHMiOlsiIl0sInJlc291cmNlcyI6WyJzZXJ2aWNlcyIsImVuZHBvaW50cyIsInBvZHMiXSwidmVyYnMiOlsiZ2V0IiwibGlzdCIsIndhdGNoIl19XX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pby92MSIsImtpbmQiOiJSb2xlQmluZGluZyIsIm1ldGFkYXRhIjp7ImNyZWF0aW9uVGltZXN0YW1wIjpudWxsLCJuYW1lIjoibmV0b2JzZXJ2LW9wZW5zaGlmdC1uZXRvYnNlcnYtb3BlcmF0b3ItcHJvbWV0aGV1cyJ9LCJyb2xlUmVmIjp7ImFwaUdyb3VwIjoicmJhYy5hdXRob3JpemF0aW9uLms4cy5pbyIsImtpbmQiOiJSb2xlIiwibmFtZSI6Im5ldG9ic2Vydi1vcGVuc2hpZnQtbmV0b2JzZXJ2LW9wZXJhdG9yLXByb21ldGhldXMifSwic3ViamVjdHMiOlt7ImtpbmQiOiJTZXJ2aWNlQWNjb3VudCIsIm5hbWUiOiJwcm9tZXRoZXVzLWs4cyIsIm5hbWVzcGFjZSI6Im9wZW5zaGlmdC1tb25pdG9yaW5nIn1dfQ== +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoidjEiLCJkYXRhIjp7ImNvbnRyb2xsZXJfbWFuYWdlcl9jb25maWcueWFtbCI6ImFwaVZlcnNpb246IGNvbnRyb2xsZXItcnVudGltZS5zaWdzLms4cy5pby92MWFscGhhMVxua2luZDogQ29udHJvbGxlck1hbmFnZXJDb25maWdcbmhlYWx0aDpcbiAgaGVhbHRoUHJvYmVCaW5kQWRkcmVzczogOjgwODFcbm1ldHJpY3M6XG4gIGJpbmRBZGRyZXNzOiA6ODQ0M1xud2ViaG9vazpcbiAgcG9ydDogOTQ0M1xubGVhZGVyRWxlY3Rpb246XG4gIGxlYWRlckVsZWN0OiB0cnVlXG4gIHJlc291cmNlTmFtZTogN2E3ZWNkY2QubmV0b2JzZXJ2LmlvXG4ifSwia2luZCI6IkNvbmZpZ01hcCIsIm1ldGFkYXRhIjp7ImxhYmVscyI6eyJhcHAiOiJuZXRvYnNlcnYtb3BlcmF0b3IifSwibmFtZSI6Im5ldG9ic2Vydi1tYW5hZ2VyLWNvbmZpZyJ9fQ== +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoidjEiLCJraW5kIjoiU2VydmljZSIsIm1ldGFkYXRhIjp7ImFubm90YXRpb25zIjp7InNlcnZpY2UuYmV0YS5vcGVuc2hpZnQuaW8vc2VydmluZy1jZXJ0LXNlY3JldC1uYW1lIjoibWFuYWdlci1tZXRyaWNzLXRscyJ9LCJjcmVhdGlvblRpbWVzdGFtcCI6bnVsbCwibGFiZWxzIjp7ImFwcCI6Im5ldG9ic2Vydi1vcGVyYXRvciIsImNvbnRyb2wtcGxhbmUiOiJjb250cm9sbGVyLW1hbmFnZXIifSwibmFtZSI6Im5ldG9ic2Vydi1tZXRyaWNzLXNlcnZpY2UifSwic3BlYyI6eyJwb3J0cyI6W3sibmFtZSI6Imh0dHBzIiwicG9ydCI6ODQ0MywicHJvdG9jb2wiOiJUQ1AiLCJ0YXJnZXRQb3J0Ijo4NDQzfV0sInNlbGVjdG9yIjp7ImFwcCI6Im5ldG9ic2Vydi1vcGVyYXRvciIsImNvbnRyb2wtcGxhbmUiOiJjb250cm9sbGVyLW1hbmFnZXIifX0sInN0YXR1cyI6eyJsb2FkQmFsYW5jZXIiOnt9fX0= +- type: olm.bundle.object + value: + data: eyJhcGlWZXJzaW9uIjoidjEiLCJraW5kIjoiU2VydmljZSIsIm1ldGFkYXRhIjp7ImFubm90YXRpb25zIjp7InNlcnZpY2UuYmV0YS5vcGVuc2hpZnQuaW8vc2VydmluZy1jZXJ0LXNlY3JldC1uYW1lIjoid2ViaG9vay1zZXJ2ZXItY2VydCJ9LCJjcmVhdGlvblRpbWVzdGFtcCI6bnVsbCwibGFiZWxzIjp7ImFwcC5rdWJlcm5ldGVzLmlvL2NvbXBvbmVudCI6IndlYmhvb2siLCJhcHAua3ViZXJuZXRlcy5pby9jcmVhdGVkLWJ5IjoibmV0b2JzZXJ2LW9wZXJhdG9yIiwiYXBwLmt1YmVybmV0ZXMuaW8vaW5zdGFuY2UiOiJ3ZWJob29rLXNlcnZpY2UiLCJhcHAua3ViZXJuZXRlcy5pby9tYW5hZ2VkLWJ5Ijoia3VzdG9taXplIiwiYXBwLmt1YmVybmV0ZXMuaW8vbmFtZSI6InNlcnZpY2UiLCJhcHAua3ViZXJuZXRlcy5pby9wYXJ0LW9mIjoibmV0b2JzZXJ2LW9wZXJhdG9yIn0sIm5hbWUiOiJuZXRvYnNlcnYtd2ViaG9vay1zZXJ2aWNlIn0sInNwZWMiOnsicG9ydHMiOlt7InBvcnQiOjQ0MywicHJvdG9jb2wiOiJUQ1AiLCJ0YXJnZXRQb3J0Ijo5NDQzfV0sInNlbGVjdG9yIjp7ImNvbnRyb2wtcGxhbmUiOiJjb250cm9sbGVyLW1hbmFnZXIifX0sInN0YXR1cyI6eyJsb2FkQmFsYW5jZXIiOnt9fX0= +relatedImages: +- image: quay.io/netobserv/flowlogs-pipeline:v1.9.2-community + name: flowlogs-pipeline +- image: quay.io/netobserv/netobserv-ebpf-agent:v1.9.2-community + name: ebpf-agent +- image: quay.io/netobserv/network-observability-console-plugin-pf4:v1.8.2-community + name: console-plugin-compat +- image: quay.io/netobserv/network-observability-console-plugin:v1.9.2-community + name: console-plugin +- image: quay.io/netobserv/network-observability-operator-bundle:v1.9.2-community + name: "" +- image: quay.io/netobserv/network-observability-operator:1.9.2-community + name: "" +schema: olm.bundle diff --git a/catalog/unreleased-legacy/v1.9.2-community/index.yaml b/catalog/unreleased-legacy/v1.9.2-community/index.yaml new file mode 100644 index 000000000..8be74e910 --- /dev/null +++ b/catalog/unreleased-legacy/v1.9.2-community/index.yaml @@ -0,0 +1,6 @@ +--- +entries: + - name: netobserv-operator.v1.9.2-community +name: latest +package: netobserv-operator +schema: olm.channel diff --git a/catalog/unreleased-legacy/v1.9.2-community/other.yaml b/catalog/unreleased-legacy/v1.9.2-community/other.yaml new file mode 100644 index 000000000..b4b1b0cf9 --- /dev/null +++ b/catalog/unreleased-legacy/v1.9.2-community/other.yaml @@ -0,0 +1,7 @@ +--- +defaultChannel: latest +icon: + base64data: PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPCEtLSBHZW5lcmF0b3I6IEFkb2JlIElsbHVzdHJhdG9yIDI2LjAuMiwgU1ZHIEV4cG9ydCBQbHVnLUluIC4gU1ZHIFZlcnNpb246IDYuMDAgQnVpbGQgMCkgIC0tPgo8c3ZnIHZlcnNpb249IjEuMSIgaWQ9IkxheWVyXzEiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHg9IjBweCIgeT0iMHB4IgoJIHZpZXdCb3g9IjAgMCAxMDAgMTAwIiBzdHlsZT0iZW5hYmxlLWJhY2tncm91bmQ6bmV3IDAgMCAxMDAgMTAwOyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+CjxzdHlsZSB0eXBlPSJ0ZXh0L2NzcyI+Cgkuc3Qwe2ZpbGw6dXJsKCNTVkdJRF8xXyk7fQoJLnN0MXtmaWxsOiNGRkZGRkY7fQoJLnN0MntvcGFjaXR5OjAuNjt9Cgkuc3Qze29wYWNpdHk6MC41O30KCS5zdDR7b3BhY2l0eTowLjQ7fQo8L3N0eWxlPgo8Zz4KCTxnPgoJCTxnPgoJCQk8cmFkaWFsR3JhZGllbnQgaWQ9IlNWR0lEXzFfIiBjeD0iMTQuNzc1OCIgY3k9Ii0yLjk3NzEiIHI9IjkxLjYyNyIgZ3JhZGllbnRVbml0cz0idXNlclNwYWNlT25Vc2UiPgoJCQkJPHN0b3AgIG9mZnNldD0iMCIgc3R5bGU9InN0b3AtY29sb3I6IzNDM0ZBNiIvPgoJCQkJPHN0b3AgIG9mZnNldD0iMSIgc3R5bGU9InN0b3AtY29sb3I6IzNCMDM0MCIvPgoJCQk8L3JhZGlhbEdyYWRpZW50PgoJCQk8cGF0aCBjbGFzcz0ic3QwIiBkPSJNNTAsOTljLTEzLjMsMC0yNS40LTUuMy0zNC4yLTEzLjlDNi43LDc2LjIsMSw2My43LDEsNTBDMSwyMi45LDIyLjksMSw1MCwxYzEzLjcsMCwyNi4yLDUuNywzNS4xLDE0LjgKCQkJCUM5My43LDI0LjYsOTksMzYuNyw5OSw1MEM5OSw3Ny4xLDc3LjEsOTksNTAsOTl6Ii8+CgkJPC9nPgoJCTxnPgoJCQk8Y2lyY2xlIGNsYXNzPSJzdDEiIGN4PSIzNy41IiBjeT0iODEuOSIgcj0iNSIvPgoJCTwvZz4KCQk8cGF0aCBjbGFzcz0ic3QxIiBkPSJNNDguNiw5MS45bDE4LjgtNDMuM2MtMi41LTAuMS01LTAuNy03LjItMkwzMy4yLDY4LjJsMS40LTEuOGwyMC0yNS4xYy0xLjUtMi40LTIuMy01LjEtMi4zLTcuOUw5LDUyLjIKCQkJbDQ3LjYtMjkuOWwwLDBjMC4xLTAuMSwwLjItMC4yLDAuMi0wLjJjNi4xLTYuMSwxNS45LTYuMSwyMiwwbDAuMSwwLjFjNiw2LjEsNiwxNS45LTAuMSwyMS45Yy0wLjEsMC4xLTAuMiwwLjItMC4yLDAuMmwwLDAKCQkJTDQ4LjYsOTEuOXoiLz4KCQk8ZyBjbGFzcz0ic3QyIj4KCQkJPGNpcmNsZSBjbGFzcz0ic3QxIiBjeD0iNTAuMyIgY3k9IjE0LjciIHI9IjMuMSIvPgoJCTwvZz4KCQk8ZyBjbGFzcz0ic3QzIj4KCQkJPGNpcmNsZSBjbGFzcz0ic3QxIiBjeD0iMjcuNyIgY3k9IjU4IiByPSIxLjciLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9Ijc3LjQiIGN5PSI2OS4zIiByPSIxLjciLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjE2LjMiIGN5PSIzNi42IiByPSIxLjciLz4KCQk8L2c+CgkJPGcgY2xhc3M9InN0NCI+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjYzLjciIGN5PSI4NS45IiByPSIyLjIiLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjI5LjQiIGN5PSIxOS42IiByPSI0LjgiLz4KCQk8L2c+CgkJPGcgY2xhc3M9InN0MyI+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9Ijg4IiBjeT0iNTAiIHI9IjQuOCIvPgoJCTwvZz4KCTwvZz4KPC9nPgo8L3N2Zz4K + mediatype: image/svg+xml +name: netobserv-operator +schema: olm.package diff --git a/catalog/unreleased/v1.9.2-community/bundle.yaml b/catalog/unreleased/v1.9.2-community/bundle.yaml new file mode 100644 index 000000000..bdadc6eb0 --- /dev/null +++ b/catalog/unreleased/v1.9.2-community/bundle.yaml @@ -0,0 +1,840 @@ +--- +image: quay.io/netobserv/network-observability-operator-bundle:v1.9.2-community +name: netobserv-operator.v1.9.2-community +package: netobserv-operator +properties: +- type: olm.gvk + value: + group: flows.netobserv.io + kind: FlowCollector + version: v1beta2 +- type: olm.gvk + value: + group: flows.netobserv.io + kind: FlowMetric + version: v1alpha1 +- type: olm.package + value: + packageName: netobserv-operator + version: 1.9.2-community +- type: olm.csv.metadata + value: + annotations: + alm-examples: |- + [ + { + "apiVersion": "flows.netobserv.io/v1alpha1", + "kind": "FlowMetric", + "metadata": { + "labels": { + "app.kubernetes.io/created-by": "netobserv-operator", + "app.kubernetes.io/instance": "flowmetric-sample", + "app.kubernetes.io/managed-by": "kustomize", + "app.kubernetes.io/name": "flowmetric", + "app.kubernetes.io/part-of": "netobserv-operator" + }, + "name": "flowmetric-sample" + }, + "spec": { + "charts": [ + { + "dashboardName": "Main", + "queries": [ + { + "legend": "", + "promQL": "sum(rate($METRIC[2m]))" + } + ], + "title": "External ingress traffic", + "type": "SingleStat", + "unit": "Bps" + }, + { + "dashboardName": "Main", + "queries": [ + { + "legend": "{{DstK8S_Namespace}} / {{DstK8S_OwnerName}}", + "promQL": "sum(rate($METRIC{DstK8S_Namespace!=\"\"}[2m])) by (DstK8S_Namespace, DstK8S_OwnerName)" + } + ], + "sectionName": "External", + "title": "Top external ingress traffic per workload", + "type": "StackArea", + "unit": "Bps" + } + ], + "direction": "Ingress", + "filters": [ + { + "field": "SrcSubnetLabel", + "matchType": "Absence" + } + ], + "labels": [ + "DstK8S_HostName", + "DstK8S_Namespace", + "DstK8S_OwnerName", + "DstK8S_OwnerType" + ], + "metricName": "cluster_external_ingress_bytes_total", + "type": "Counter", + "valueField": "Bytes" + } + }, + { + "apiVersion": "flows.netobserv.io/v1beta2", + "kind": "FlowCollector", + "metadata": { + "name": "cluster" + }, + "spec": { + "agent": { + "ebpf": { + "cacheActiveTimeout": "5s", + "cacheMaxFlows": 100000, + "excludeInterfaces": [ + "lo" + ], + "imagePullPolicy": "IfNotPresent", + "interfaces": [], + "kafkaBatchSize": 1048576, + "logLevel": "info", + "metrics": { + "server": { + "port": 9400 + } + }, + "privileged": false, + "resources": { + "limits": { + "memory": "800Mi" + }, + "requests": { + "cpu": "100m", + "memory": "50Mi" + } + }, + "sampling": 50 + }, + "type": "eBPF" + }, + "consolePlugin": { + "autoscaler": { + "maxReplicas": 3, + "metrics": [ + { + "resource": { + "name": "cpu", + "target": { + "averageUtilization": 50, + "type": "Utilization" + } + }, + "type": "Resource" + } + ], + "minReplicas": 1, + "status": "Disabled" + }, + "enable": true, + "imagePullPolicy": "IfNotPresent", + "logLevel": "info", + "portNaming": { + "enable": true, + "portNames": { + "3100": "loki" + } + }, + "quickFilters": [ + { + "default": true, + "filter": { + "flow_layer": "\"app\"" + }, + "name": "Applications" + }, + { + "filter": { + "flow_layer": "\"infra\"" + }, + "name": "Infrastructure" + }, + { + "default": true, + "filter": { + "dst_kind": "\"Pod\"", + "src_kind": "\"Pod\"" + }, + "name": "Pods network" + }, + { + "filter": { + "dst_kind": "\"Service\"" + }, + "name": "Services network" + } + ], + "replicas": 1, + "resources": { + "limits": { + "memory": "100Mi" + }, + "requests": { + "cpu": "100m", + "memory": "50Mi" + } + } + }, + "deploymentModel": "Direct", + "exporters": [], + "kafka": { + "address": "kafka-cluster-kafka-bootstrap.netobserv", + "tls": { + "caCert": { + "certFile": "ca.crt", + "name": "kafka-cluster-cluster-ca-cert", + "type": "secret" + }, + "enable": false, + "userCert": { + "certFile": "user.crt", + "certKey": "user.key", + "name": "flp-kafka", + "type": "secret" + } + }, + "topic": "network-flows" + }, + "loki": { + "enable": true, + "lokiStack": { + "name": "loki" + }, + "mode": "Monolithic", + "monolithic": { + "tenantID": "netobserv", + "tls": { + "caCert": { + "certFile": "service-ca.crt", + "name": "loki-gateway-ca-bundle", + "type": "configmap" + }, + "enable": false + }, + "url": "http://loki.netobserv.svc:3100/" + }, + "readTimeout": "30s", + "writeBatchSize": 10485760, + "writeBatchWait": "1s", + "writeTimeout": "10s" + }, + "namespace": "netobserv", + "networkPolicy": { + "additionalNamespaces": [], + "enable": false + }, + "processor": { + "imagePullPolicy": "IfNotPresent", + "kafkaConsumerAutoscaler": null, + "kafkaConsumerBatchSize": 10485760, + "kafkaConsumerQueueCapacity": 1000, + "kafkaConsumerReplicas": 3, + "logLevel": "info", + "logTypes": "Flows", + "metrics": { + "disableAlerts": [], + "server": { + "port": 9401 + } + }, + "resources": { + "limits": { + "memory": "800Mi" + }, + "requests": { + "cpu": "100m", + "memory": "100Mi" + } + } + }, + "prometheus": { + "querier": { + "enable": true, + "mode": "Auto", + "timeout": "30s" + } + } + } + } + ] + capabilities: Seamless Upgrades + categories: Monitoring, Networking, Observability + console.openshift.io/plugins: '["netobserv-plugin"]' + containerImage: quay.io/netobserv/network-observability-operator:1.9.2-community + createdAt: "2025-09-08T08:23:26Z" + description: Network flows collector and monitoring solution + operatorframework.io/initialization-resource: '{"apiVersion":"flows.netobserv.io/v1beta2", + "kind":"FlowCollector","metadata":{"name":"cluster"},"spec": {}}' + operatorframework.io/suggested-namespace: openshift-netobserv-operator + operators.operatorframework.io/builder: operator-sdk-v1.40.0 + operators.operatorframework.io/project_layout: go.kubebuilder.io/v4 + repository: https://github.com/netobserv/network-observability-operator + support: NetObserv team + apiServiceDefinitions: {} + crdDescriptions: + owned: + - description: '`FlowCollector` is the schema for the network flows collection + API, which pilots and configures the underlying deployments.' + displayName: Flow Collector + kind: FlowCollector + name: flowcollectors.flows.netobserv.io + specDescriptors: + - description: defines the desired type of deployment for flow processing. + displayName: Deployment model + path: deploymentModel + - description: for flows extraction. + displayName: Agent configuration + path: agent + - path: agent.type + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - path: agent.ipfix + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - description: Settings related to the eBPF-based flow reporter. + displayName: eBPF Agent configuration + path: agent.ebpf + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:agent.type:eBPF + - displayName: Privileged mode + path: agent.ebpf.privileged + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - displayName: Cache active timeout + path: agent.ebpf.cacheActiveTimeout + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Cache max flows + path: agent.ebpf.cacheMaxFlows + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Kafka batch size + path: agent.ebpf.kafkaBatchSize + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Log level + path: agent.ebpf.logLevel + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Image pull policy + path: agent.ebpf.imagePullPolicy + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:imagePullPolicy + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Resource Requirements + path: agent.ebpf.resources + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:resourceRequirements + - path: agent.ebpf.advanced + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - path: agent.ebpf.flowFilter + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - path: agent.ebpf.metrics.enable + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - description: to use Kafka as a broker as part of the flow collection pipeline. + displayName: Kafka configuration + path: kafka + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka + - displayName: TLS configuration + path: kafka.tls + - path: kafka.tls.enable + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - displayName: Insecure + path: kafka.tls.insecureSkipVerify + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:kafka.tls.enable:true + - displayName: User certificate when using mTLS + path: kafka.tls.userCert + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:kafka.tls.enable:true + - displayName: CA certificate + path: kafka.tls.caCert + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:kafka.tls.enable:true + - path: kafka.sasl + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - description: of the component that receives the flows from the agent, enriches + them, generates metrics, and forwards them to the Loki persistence layer + and/or any available exporter. + displayName: Processor configuration + path: processor + - displayName: Multi-cluster deployment + path: processor.multiClusterDeployment + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - displayName: Cluster name + path: processor.clusterName + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.multiClusterDeployment:true + - displayName: Availability zones + path: processor.addZone + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - path: processor.advanced + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - displayName: Metrics configuration + path: processor.metrics + - displayName: Server configuration + path: processor.metrics.server + - displayName: TLS configuration + path: processor.metrics.server.tls + - displayName: Insecure + path: processor.metrics.server.tls.insecureSkipVerify + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.metrics.server.tls.type:Provided + - displayName: Cert + path: processor.metrics.server.tls.provided + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.metrics.server.tls.type:Provided + - displayName: CA + path: processor.metrics.server.tls.providedCaFile + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:processor.metrics.server.tls.type:Provided + - displayName: Kafka consumer replicas + path: processor.kafkaConsumerReplicas + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: kafka consumer autoscaler + path: processor.kafkaConsumerAutoscaler + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Kafka consumer queue capacity + path: processor.kafkaConsumerQueueCapacity + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Kafka consumer batch size + path: processor.kafkaConsumerBatchSize + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:deploymentModel:Kafka + - urn:alm:descriptor:com.tectonic.ui:advanced + - path: processor.subnetLabels.openShiftAutoDetect + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - displayName: Log level + path: processor.logLevel + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Image pull policy + path: processor.imagePullPolicy + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:imagePullPolicy + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Resource Requirements + path: processor.resources + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:resourceRequirements + - description: for the flow store. + displayName: Loki client settings + path: loki + - displayName: Enable + path: loki.enable + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - displayName: Mode + path: loki.mode + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true + - displayName: Loki stack + path: loki.lokiStack + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:LokiStack + - displayName: Monolithic + path: loki.monolithic + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:Monolithic + - displayName: Microservices + path: loki.microservices + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:Microservices + - displayName: Manual + path: loki.manual + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.mode:Manual + - displayName: Write batch wait + path: loki.writeBatchWait + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Write batch size + path: loki.writeBatchSize + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Write timeout + path: loki.writeTimeout + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - path: loki.advanced + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - description: related to the OpenShift Console integration. + displayName: Console plugin configuration + path: consolePlugin + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true + - displayName: Enable + path: consolePlugin.enable + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:booleanSwitch + - displayName: Port naming + path: consolePlugin.portNaming + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - displayName: Quick filters + path: consolePlugin.quickFilters + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - displayName: Replicas + path: consolePlugin.replicas + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Horizontal pod autoscaler + path: consolePlugin.autoscaler + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Log level + path: consolePlugin.logLevel + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Image pull policy + path: consolePlugin.imagePullPolicy + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:imagePullPolicy" + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Resource Requirements + path: consolePlugin.resources + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:resourceRequirements + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:consolePlugin.enable:true + - path: consolePlugin.advanced + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:hidden + - description: additional optional exporters for custom consumption or storage. + displayName: Exporters + path: exporters + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Type + path: exporters[0].type + - displayName: IPFIX configuration + path: exporters[0].ipfix + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:exporters.type:IPFIX + - displayName: Kafka configuration + path: exporters[0].kafka + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:exporters.type:Kafka + - displayName: OpenTelemetry configuration + path: exporters[0].openTelemetry + x-descriptors: + - urn:alm:descriptor:com.tectonic.ui:fieldDependency:exporters.type:OpenTelemetry + - displayName: Exclude interfaces + path: agent.ebpf.excludeInterfaces + - displayName: Features + path: agent.ebpf.features + - displayName: Interfaces + path: agent.ebpf.interfaces + - displayName: Metrics + path: agent.ebpf.metrics + - displayName: Disable alerts + path: agent.ebpf.metrics.disableAlerts + - displayName: Server + path: agent.ebpf.metrics.server + - displayName: Port + path: agent.ebpf.metrics.server.port + - displayName: Sampling + path: agent.ebpf.sampling + - displayName: Enable + path: consolePlugin.portNaming.enable + - displayName: Port names + path: consolePlugin.portNaming.portNames + - displayName: Address + path: kafka.address + - displayName: Topic + path: kafka.topic + - displayName: Name + path: loki.lokiStack.name + - displayName: Namespace + path: loki.lokiStack.namespace + - displayName: Auth token + path: loki.manual.authToken + - displayName: Ingester url + path: loki.manual.ingesterUrl + - displayName: Querier url + path: loki.manual.querierUrl + - displayName: Status url + path: loki.manual.statusUrl + - displayName: Tenant id + path: loki.manual.tenantID + - displayName: Ingester url + path: loki.microservices.ingesterUrl + - displayName: Querier url + path: loki.microservices.querierUrl + - displayName: Tenant id + path: loki.microservices.tenantID + - displayName: Tenant id + path: loki.monolithic.tenantID + - displayName: Url + path: loki.monolithic.url + - displayName: Read timeout + path: loki.readTimeout + - displayName: Namespace + path: namespace + - displayName: Network policy + path: networkPolicy + - displayName: Additional namespaces + path: networkPolicy.additionalNamespaces + - displayName: Enable + path: networkPolicy.enable + - displayName: Deduper + path: processor.deduper + - displayName: Mode + path: processor.deduper.mode + - displayName: Sampling + path: processor.deduper.sampling + - displayName: Filters + path: processor.filters + - displayName: Log types + path: processor.logTypes + - displayName: Disable alerts + path: processor.metrics.disableAlerts + - displayName: Include list + path: processor.metrics.includeList + - displayName: Port + path: processor.metrics.server.port + - displayName: Subnet labels + path: processor.subnetLabels + - displayName: Custom labels + path: processor.subnetLabels.customLabels + - displayName: Prometheus + path: prometheus + - displayName: Querier + path: prometheus.querier + - displayName: Enable + path: prometheus.querier.enable + - displayName: Manual + path: prometheus.querier.manual + - displayName: Forward user token + path: prometheus.querier.manual.forwardUserToken + - displayName: Url + path: prometheus.querier.manual.url + - displayName: Mode + path: prometheus.querier.mode + - displayName: Timeout + path: prometheus.querier.timeout + statusDescriptors: + - description: Namespace where console plugin and flowlogs-pipeline have been + deployed. + displayName: Namespace + path: namespace + x-descriptors: + - urn:alm:descriptor:text + - description: Conditions of the FlowCollector instance health. + displayName: Conditions + path: conditions + x-descriptors: + - urn:alm:descriptor:io.kubernetes.conditions + version: v1beta2 + - description: '`FlowMetric` is the schema for the custom metrics API, which + allows to generate more metrics out of flow logs. You can find examples + here: https://github.com/netobserv/network-observability-operator/tree/main/config/samples/flowmetrics' + displayName: Flow Metric + kind: FlowMetric + name: flowmetrics.flows.netobserv.io + version: v1alpha1 + description: |- + NetObserv Operator is an OpenShift / Kubernetes operator for network observability. It deploys a monitoring pipeline that consists in: + - an eBPF agent, that generates network flows from captured packets + - flowlogs-pipeline, a component that collects, enriches and exports these flows + - when used in OpenShift, a Console plugin for flows visualization with powerful filtering options, a topology representation and more + + Flow data is then available in multiple ways, each optional: + + - As Prometheus metrics + - As raw flow logs stored in Grafana Loki + - As raw flow logs exported to a collector + + ## Dependencies + + ### Loki + + [Loki](https://grafana.com/oss/loki/), from GrafanaLabs, can optionally be used as the backend to store all collected flows. The NetObserv Operator does not install Loki directly, however we provide some guidance to help you there. + + For normal usage, we recommend two options: + + - Installing the [Loki Operator](https://loki-operator.dev/docs/prologue/quickstart.md/). We have written [a guide](https://github.com/netobserv/documents/blob/main/loki_operator.md) to help you through those steps. Please note that it requires configuring an object storage. Note also that the Loki Operator can also be used for [OpenShift cluster logging](https://docs.openshift.com/container-platform/latest/logging/cluster-logging.html). If you do so, you should not share the same `LokiStack` for Logging and NetObserv. + + - Installing using [Grafana's official documentation](https://grafana.com/docs/loki/latest/). Here also we wrote a ["distributed Loki" step by step guide](https://github.com/netobserv/documents/blob/main/loki_distributed.md). + + For a quick try that is not suitable for production and not scalable (it deploys a single pod, configures a 10GB storage PVC, with 24 hours of retention), you can simply run the following commands: + + ``` + kubectl create namespace netobserv + kubectl apply -f <(curl -L https://raw.githubusercontent.com/netobserv/documents/5410e65b8e05aaabd1244a9524cfedd8ac8c56b5/examples/zero-click-loki/1-storage.yaml) -n netobserv + kubectl apply -f <(curl -L https://raw.githubusercontent.com/netobserv/documents/5410e65b8e05aaabd1244a9524cfedd8ac8c56b5/examples/zero-click-loki/2-loki.yaml) -n netobserv + ``` + + If you prefer to not use Loki, you must set `spec.loki.enable` to `false` in `FlowCollector`. + In that case, you can still get the Prometheus metrics or export raw flows to a custom collector. But be aware that some of the Console plugin features will be disabled. For instance, you will not be able to view raw flows there, and the metrics / topology will have a more limited level of details, missing information such as pods or IPs. + + ### Kafka + + [Apache Kafka](https://kafka.apache.org/) can optionally be used for a more resilient and scalable architecture. You can use for example [Strimzi](https://strimzi.io/), an operator-based distribution of Kafka for Kubernetes and OpenShift. + + ### Grafana + + [Grafana](https://grafana.com/oss/grafana/) can optionally be installed for custom dashboards and query capabilities. + + ## Configuration + + The `FlowCollector` resource is used to configure the operator and its managed components. A comprehensive documentation is [available here](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/docs/FlowCollector.md), and a full sample file [there](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/config/samples/flows_v1beta2_flowcollector.yaml). + + To edit configuration in cluster, run: + + ```bash + kubectl edit flowcollector cluster + ``` + + As it operates cluster-wide on every node, only a single `FlowCollector` is allowed, and it has to be named `cluster`. + + A couple of settings deserve special attention: + + - Sampling (`spec.agent.ebpf.sampling`): a value of `100` means: one flow every 100 is sampled. `1` means all flows are sampled. The lower it is, the more flows you get, and the more accurate are derived metrics, but the higher amount of resources are consumed. By default, sampling is set to 50 (ie. 1:50). Note that more sampled flows also means more storage needed. We recommend to start with default values and refine empirically, to figure out which setting your cluster can manage. + + - Loki (`spec.loki`): configure here how to reach Loki. The default values match the Loki quick install paths mentioned above, but you might have to configure differently if you used another installation method. Make sure to disable it (`spec.loki.enable`) if you don't want to use Loki. + + - Quick filters (`spec.consolePlugin.quickFilters`): configure preset filters to be displayed in the Console plugin. They offer a way to quickly switch from filters to others, such as showing / hiding pods network, or infrastructure network, or application network, etc. They can be tuned to reflect the different workloads running on your cluster. For a list of available filters, [check this page](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/docs/QuickFilters.md). + + - Kafka (`spec.deploymentModel: KAFKA` and `spec.kafka`): when enabled, integrates the flow collection pipeline with Kafka, by splitting ingestion from transformation (kube enrichment, derived metrics, ...). Kafka can provide better scalability, resiliency and high availability ([view more details](https://www.redhat.com/en/topics/integration/what-is-apache-kafka)). Assumes Kafka is already deployed and a topic is created. + + - Exporters (`spec.exporters`) an optional list of exporters to which to send enriched flows. KAFKA and IPFIX exporters are supported. This allows you to define any custom storage or processing that can read from Kafka or use the IPFIX standard. + + - To enable availability zones awareness, set `spec.processor.addZone` to `true`. + + ## Resource considerations + + The following table outlines examples of resource considerations for clusters with certain workload sizes. + The examples outlined in the table demonstrate scenarios that are tailored to specific workloads. Consider each example only as a baseline from which adjustments can be made to accommodate your workload needs. + + + | Resource recommendations | Extra small (10 nodes) | Small (25 nodes) | Medium (65 nodes) ** | Large (120 nodes) ** | + | ----------------------------------------------- | ---------------------- | ---------------------- | ----------------------- | ----------------------------- | + | *Worker Node vCPU and memory* | 4 vCPUs\| 16GiB mem * | 16 vCPUs\| 64GiB mem * | 16 vCPUs\| 64GiB mem * |16 vCPUs\| 64GiB Mem * | + | *LokiStack size* | `1x.extra-small` | `1x.small` | `1x.small` | `1x.medium` | + | *Network Observability controller memory limit* | 400Mi (default) | 400Mi (default) | 400Mi (default) | 800Mi | + | *eBPF sampling rate* | 50 (default) | 50 (default) | 50 (default) | 50 (default) | + | *eBPF memory limit* | 800Mi (default) | 800Mi (default) | 2000Mi | 800Mi (default) | + | *FLP memory limit* | 800Mi (default) | 800Mi (default) | 800Mi (default) | 800Mi (default) | + | *FLP Kafka partitions* | N/A | 48 | 48 | 48 | + | *Kafka consumer replicas* | N/A | 24 | 24 | 24 | + | *Kafka brokers* | N/A | 3 (default) | 3 (default) | 3 (default) | + + *. Tested with AWS M6i instances. + **. In addition to this worker and its controller, 3 infra nodes (size `M6i.12xlarge`) and 1 workload node (size `M6i.8xlarge`) were tested. + + ## Further reading + + Please refer to the documentation on GitHub for more information. + + This documentation includes: + + - An [overview](https://github.com/netobserv/network-observability-operator#openshift-console) of the features, with screenshots + - More information on [configuring metrics](https://github.com/netobserv/network-observability-operator/blob/1.9.2-community/docs/Metrics.md). + - A [performance](https://github.com/netobserv/network-observability-operator#performance-fine-tuning) section, for fine-tuning + - A [security](https://github.com/netobserv/network-observability-operator#securing-data-and-communications) section + - An [F.A.Q.](https://github.com/netobserv/network-observability-operator#faq--troubleshooting) section + displayName: NetObserv Operator + installModes: + - supported: false + type: OwnNamespace + - supported: false + type: SingleNamespace + - supported: false + type: MultiNamespace + - supported: true + type: AllNamespaces + keywords: + - network observability + - ebpf + - ipfix + - flow tracing + - flows + - topology + - network + - observability + labels: + operatorframework.io/arch.amd64: supported + operatorframework.io/arch.arm64: supported + operatorframework.io/arch.ppc64le: supported + operatorframework.io/arch.s390x: supported + operatorframework.io/os.linux: supported + links: + - name: Project page + url: https://github.com/netobserv/network-observability-operator + - name: Issue tracker + url: https://github.com/netobserv/network-observability-operator/issues + - name: Discussion board + url: https://github.com/netobserv/network-observability-operator/discussions + maintainers: + - email: jpinsonn@redhat.com + name: Julien Pinsonneau + - email: jtakvori@redhat.com + name: Joel Takvorian + - email: kmeth@redhat.com + name: Kalman Meth + - email: mmahmoud@redhat.com + name: Mohamed S. Mahmoud + - email: ocazade@redhat.com + name: Olivier Cazade + - email: rschaffe@redhat.com + name: Ronen Schaffer + - email: stlee@redhat.com + name: Steven Lee + maturity: alpha + minKubeVersion: 1.23.0 + provider: + name: Red Hat + url: https://www.redhat.com +relatedImages: +- image: quay.io/netobserv/flowlogs-pipeline:v1.9.2-community + name: flowlogs-pipeline +- image: quay.io/netobserv/netobserv-ebpf-agent:v1.9.2-community + name: ebpf-agent +- image: quay.io/netobserv/network-observability-console-plugin-pf4:v1.8.2-community + name: console-plugin-compat +- image: quay.io/netobserv/network-observability-console-plugin:v1.9.2-community + name: console-plugin +- image: quay.io/netobserv/network-observability-operator-bundle:v1.9.2-community + name: "" +- image: quay.io/netobserv/network-observability-operator:1.9.2-community + name: "" +schema: olm.bundle diff --git a/catalog/unreleased/v1.9.2-community/index.yaml b/catalog/unreleased/v1.9.2-community/index.yaml new file mode 100644 index 000000000..8be74e910 --- /dev/null +++ b/catalog/unreleased/v1.9.2-community/index.yaml @@ -0,0 +1,6 @@ +--- +entries: + - name: netobserv-operator.v1.9.2-community +name: latest +package: netobserv-operator +schema: olm.channel diff --git a/catalog/unreleased/v1.9.2-community/other.yaml b/catalog/unreleased/v1.9.2-community/other.yaml new file mode 100644 index 000000000..b4b1b0cf9 --- /dev/null +++ b/catalog/unreleased/v1.9.2-community/other.yaml @@ -0,0 +1,7 @@ +--- +defaultChannel: latest +icon: + base64data: PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPCEtLSBHZW5lcmF0b3I6IEFkb2JlIElsbHVzdHJhdG9yIDI2LjAuMiwgU1ZHIEV4cG9ydCBQbHVnLUluIC4gU1ZHIFZlcnNpb246IDYuMDAgQnVpbGQgMCkgIC0tPgo8c3ZnIHZlcnNpb249IjEuMSIgaWQ9IkxheWVyXzEiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM6eGxpbms9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsiIHg9IjBweCIgeT0iMHB4IgoJIHZpZXdCb3g9IjAgMCAxMDAgMTAwIiBzdHlsZT0iZW5hYmxlLWJhY2tncm91bmQ6bmV3IDAgMCAxMDAgMTAwOyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+CjxzdHlsZSB0eXBlPSJ0ZXh0L2NzcyI+Cgkuc3Qwe2ZpbGw6dXJsKCNTVkdJRF8xXyk7fQoJLnN0MXtmaWxsOiNGRkZGRkY7fQoJLnN0MntvcGFjaXR5OjAuNjt9Cgkuc3Qze29wYWNpdHk6MC41O30KCS5zdDR7b3BhY2l0eTowLjQ7fQo8L3N0eWxlPgo8Zz4KCTxnPgoJCTxnPgoJCQk8cmFkaWFsR3JhZGllbnQgaWQ9IlNWR0lEXzFfIiBjeD0iMTQuNzc1OCIgY3k9Ii0yLjk3NzEiIHI9IjkxLjYyNyIgZ3JhZGllbnRVbml0cz0idXNlclNwYWNlT25Vc2UiPgoJCQkJPHN0b3AgIG9mZnNldD0iMCIgc3R5bGU9InN0b3AtY29sb3I6IzNDM0ZBNiIvPgoJCQkJPHN0b3AgIG9mZnNldD0iMSIgc3R5bGU9InN0b3AtY29sb3I6IzNCMDM0MCIvPgoJCQk8L3JhZGlhbEdyYWRpZW50PgoJCQk8cGF0aCBjbGFzcz0ic3QwIiBkPSJNNTAsOTljLTEzLjMsMC0yNS40LTUuMy0zNC4yLTEzLjlDNi43LDc2LjIsMSw2My43LDEsNTBDMSwyMi45LDIyLjksMSw1MCwxYzEzLjcsMCwyNi4yLDUuNywzNS4xLDE0LjgKCQkJCUM5My43LDI0LjYsOTksMzYuNyw5OSw1MEM5OSw3Ny4xLDc3LjEsOTksNTAsOTl6Ii8+CgkJPC9nPgoJCTxnPgoJCQk8Y2lyY2xlIGNsYXNzPSJzdDEiIGN4PSIzNy41IiBjeT0iODEuOSIgcj0iNSIvPgoJCTwvZz4KCQk8cGF0aCBjbGFzcz0ic3QxIiBkPSJNNDguNiw5MS45bDE4LjgtNDMuM2MtMi41LTAuMS01LTAuNy03LjItMkwzMy4yLDY4LjJsMS40LTEuOGwyMC0yNS4xYy0xLjUtMi40LTIuMy01LjEtMi4zLTcuOUw5LDUyLjIKCQkJbDQ3LjYtMjkuOWwwLDBjMC4xLTAuMSwwLjItMC4yLDAuMi0wLjJjNi4xLTYuMSwxNS45LTYuMSwyMiwwbDAuMSwwLjFjNiw2LjEsNiwxNS45LTAuMSwyMS45Yy0wLjEsMC4xLTAuMiwwLjItMC4yLDAuMmwwLDAKCQkJTDQ4LjYsOTEuOXoiLz4KCQk8ZyBjbGFzcz0ic3QyIj4KCQkJPGNpcmNsZSBjbGFzcz0ic3QxIiBjeD0iNTAuMyIgY3k9IjE0LjciIHI9IjMuMSIvPgoJCTwvZz4KCQk8ZyBjbGFzcz0ic3QzIj4KCQkJPGNpcmNsZSBjbGFzcz0ic3QxIiBjeD0iMjcuNyIgY3k9IjU4IiByPSIxLjciLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9Ijc3LjQiIGN5PSI2OS4zIiByPSIxLjciLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjE2LjMiIGN5PSIzNi42IiByPSIxLjciLz4KCQk8L2c+CgkJPGcgY2xhc3M9InN0NCI+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjYzLjciIGN5PSI4NS45IiByPSIyLjIiLz4KCQk8L2c+CgkJPGc+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9IjI5LjQiIGN5PSIxOS42IiByPSI0LjgiLz4KCQk8L2c+CgkJPGcgY2xhc3M9InN0MyI+CgkJCTxjaXJjbGUgY2xhc3M9InN0MSIgY3g9Ijg4IiBjeT0iNTAiIHI9IjQuOCIvPgoJCTwvZz4KCTwvZz4KPC9nPgo8L3N2Zz4K + mediatype: image/svg+xml +name: netobserv-operator +schema: olm.package diff --git a/config/csv/bases/netobserv-operator.clusterserviceversion.yaml b/config/csv/bases/netobserv-operator.clusterserviceversion.yaml index 080a2400a..1348dd02e 100644 --- a/config/csv/bases/netobserv-operator.clusterserviceversion.yaml +++ b/config/csv/bases/netobserv-operator.clusterserviceversion.yaml @@ -216,6 +216,8 @@ spec: x-descriptors: - urn:alm:descriptor:com.tectonic.ui:fieldDependency:loki.enable:true - urn:alm:descriptor:com.tectonic.ui:advanced + - displayName: Read timeout + path: loki.readTimeout - path: loki.advanced x-descriptors: - urn:alm:descriptor:com.tectonic.ui:hidden @@ -292,7 +294,7 @@ spec: - urn:alm:descriptor:io.kubernetes.conditions - description: '`FlowMetric` is the schema for the custom metrics API, which allows to generate more metrics out of flow logs. - You can find examples here: https://github.com/netobserv/network-observability-operator/tree/main/config/samples/flowmetrics' + You can find examples here: https://github.com/netobserv/network-observability-operator/tree/v1.9.2-community/config/samples/flowmetrics' displayName: Flow Metric kind: FlowMetric name: flowmetrics.flows.netobserv.io diff --git a/go.mod b/go.mod index 6780e2bd3..76db49222 100644 --- a/go.mod +++ b/go.mod @@ -119,3 +119,5 @@ require ( sigs.k8s.io/randfill v1.0.0 // indirect sigs.k8s.io/structured-merge-diff/v6 v6.3.0 // indirect ) + +replace github.com/netobserv/flowlogs-pipeline => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc diff --git a/go.sum b/go.sum index a81982a3f..ff29b641d 100644 --- a/go.sum +++ b/go.sum @@ -101,6 +101,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= +github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc h1:0O0vz3oHT1pj2i3dDTeIlW1LrqrmbGKw711WAjpjBoM= +github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc/go.mod h1:JKatcqNUQeNfSGMpQ73a6tsbmw/ztOZunqOYfNJ7Opg= github.com/mailru/easyjson v0.9.0 h1:PrnmzHw7262yW8sTBwxi1PdJA3Iw/EKBa8psRf7d9a4= github.com/mailru/easyjson v0.9.0/go.mod h1:1+xMtQp2MRNVL/V1bOzuP3aP8VNwRW55fQUto+XFtTU= github.com/maruel/natural v1.1.1 h1:Hja7XhhmvEFhcByqDoHz9QZbkWey+COd9xWfCfn1ioo= @@ -151,23 +153,23 @@ github.com/prometheus/procfs v0.17.0/go.mod h1:oPQLaDAMRbA+u8H5Pbfq+dl3VDAvHxMUO github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR38lUII= github.com/rogpeppe/go-internal v1.13.1/go.mod h1:uMEvuHeurkdAXX61udpOXGD/AzZDWNMNyH2VO9fmH0o= github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= -github.com/sagikazarmark/locafero v0.11.0 h1:1iurJgmM9G3PA/I+wWYIOw/5SyBtxapeHDcg+AAIFXc= -github.com/sagikazarmark/locafero v0.11.0/go.mod h1:nVIGvgyzw595SUSUE6tvCp3YYTeHs15MvlmU87WwIik= +github.com/sagikazarmark/locafero v0.9.0 h1:GbgQGNtTrEmddYDSAH9QLRyfAHY12md+8YFTqyMTC9k= +github.com/sagikazarmark/locafero v0.9.0/go.mod h1:UBUyz37V+EdMS3hDF3QWIiVr/2dPrx49OMO0Bn0hJqk= github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= -github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 h1:+jumHNA0Wrelhe64i8F6HNlS8pkoyMv5sreGx2Ry5Rw= -github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8/go.mod h1:3n1Cwaq1E1/1lhQhtRK2ts/ZwZEhjcQeJQ1RuC6Q/8U= -github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I= -github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg= -github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY= -github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo= +github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo= +github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0= +github.com/spf13/afero v1.14.0 h1:9tH6MapGnn/j0eb0yIXiLjERO8RB6xIVZRDCX7PtqWA= +github.com/spf13/afero v1.14.0/go.mod h1:acJQ8t0ohCGuMN3O+Pv0V0hgMxNYDlvdk+VTfyZmbYo= +github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y= +github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo= github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0= github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk= github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= -github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU= -github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY= +github.com/spf13/viper v1.20.1 h1:ZMi+z/lvLyPSCoNtFCpqjy0S4kPbirhpTMwl8BkW9X4= +github.com/spf13/viper v1.20.1/go.mod h1:P9Mdzt1zoHIG8m2eZQinpiBjo6kCmZSKBClNNqjJvu4= github.com/stoewer/go-strcase v1.3.1 h1:iS0MdW+kVTxgMoE1LAZyMiYJFKlOzLooE4MxjirtkAs= github.com/stoewer/go-strcase v1.3.1/go.mod h1:fAH5hQ5pehh+j3nZfvwdk2RgEgQjAoM8wodgtPmh1xo= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= diff --git a/internal/controller/consoleplugin/consoleplugin_objects.go b/internal/controller/consoleplugin/consoleplugin_objects.go index c11884af4..9fb8868cb 100644 --- a/internal/controller/consoleplugin/consoleplugin_objects.go +++ b/internal/controller/consoleplugin/consoleplugin_objects.go @@ -348,11 +348,27 @@ func (b *builder) getLokiConfig() (cfg.LokiConfig, error) { if b.desired.Loki.ReadTimeout != nil { lconf.Timeout = api.Duration{Duration: b.desired.Loki.ReadTimeout.Duration} } - if lk.TLS.Enable { - if lk.TLS.InsecureSkipVerify { + // Console plugin should always use HTTP/gateway TLS config for LokiStack mode + // regardless of clientType, since console plugin never uses gRPC + tlsConfig := &lk.TLS + if b.desired.Loki.Mode == flowslatest.LokiModeLokiStack && b.desired.Processor.HasExperimentalLokiGRPCClientProtocol() { + // Create HTTP/gateway TLS config for console plugin + tlsConfig = &flowslatest.ClientTLS{ + Enable: true, + CACert: flowslatest.CertificateReference{ + Type: flowslatest.RefTypeConfigMap, + Name: fmt.Sprintf("%s-gateway-ca-bundle", b.desired.Loki.LokiStack.Name), + Namespace: b.desired.Loki.LokiStack.Namespace, + CertFile: "service-ca.crt", + }, + } + } + + if tlsConfig.Enable { + if tlsConfig.InsecureSkipVerify { lconf.SkipTLS = true } else { - caPath := b.volumes.AddCACertificate(&lk.TLS, "loki-certs") + caPath := b.volumes.AddCACertificate(tlsConfig, "loki-certs") if caPath != "" { lconf.CAPath = caPath } diff --git a/internal/controller/consoleplugin/consoleplugin_test.go b/internal/controller/consoleplugin/consoleplugin_test.go index 026e0be99..4c4043cd8 100644 --- a/internal/controller/consoleplugin/consoleplugin_test.go +++ b/internal/controller/consoleplugin/consoleplugin_test.go @@ -283,7 +283,7 @@ func TestConfigMapUpdateWithLokistackMode(t *testing.T) { Mode: flowslatest.LokiModeLokiStack, LokiStack: flowslatest.LokiStackRef{Name: "lokistack", Namespace: "ls-namespace"}, } - loki := helper.NewLokiConfig(&lokiSpec, "any") + loki := helper.NewLokiConfig(&lokiSpec, "any", false) spec := flowslatest.FlowCollectorSpec{ConsolePlugin: plugin, Loki: lokiSpec} builder := getBuilder(&spec, &loki) old, _, _ := builder.configMap(context.Background()) @@ -292,7 +292,7 @@ func TestConfigMapUpdateWithLokistackMode(t *testing.T) { // update lokistack name lokiSpec.LokiStack.Name = "lokistack-updated" - loki = helper.NewLokiConfig(&lokiSpec, "any") + loki = helper.NewLokiConfig(&lokiSpec, "any", false) spec = flowslatest.FlowCollectorSpec{ConsolePlugin: plugin, Loki: lokiSpec} builder = getBuilder(&spec, &loki) @@ -302,7 +302,7 @@ func TestConfigMapUpdateWithLokistackMode(t *testing.T) { // update lokistack namespace lokiSpec.LokiStack.Namespace = "ls-namespace-updated" - loki = helper.NewLokiConfig(&lokiSpec, "any") + loki = helper.NewLokiConfig(&lokiSpec, "any", false) spec = flowslatest.FlowCollectorSpec{ConsolePlugin: plugin, Loki: lokiSpec} builder = getBuilder(&spec, &loki) @@ -323,7 +323,7 @@ func TestConfigMapContent(t *testing.T) { Mode: flowslatest.LokiModeLokiStack, LokiStack: flowslatest.LokiStackRef{Name: "lokistack", Namespace: "ls-namespace"}, } - loki := helper.NewLokiConfig(&lokiSpec, "any") + loki := helper.NewLokiConfig(&lokiSpec, "any", false) spec := flowslatest.FlowCollectorSpec{ Agent: agentSpec, ConsolePlugin: getPluginConfig(), diff --git a/internal/controller/flowcollector_controller.go b/internal/controller/flowcollector_controller.go index 24d1371bf..44660b764 100644 --- a/internal/controller/flowcollector_controller.go +++ b/internal/controller/flowcollector_controller.go @@ -120,7 +120,7 @@ func (r *FlowCollectorReconciler) Reconcile(ctx context.Context, _ ctrl.Request) func (r *FlowCollectorReconciler) reconcile(ctx context.Context, clh *helper.Client, desired *flowslatest.FlowCollector) error { ns := desired.Spec.GetNamespace() previousNamespace := r.status.GetDeployedNamespace(desired) - loki := helper.NewLokiConfig(&desired.Spec.Loki, ns) + loki := helper.NewLokiConfig(&desired.Spec.Loki, ns, desired.Spec.Processor.HasExperimentalLokiGRPCClientProtocol()) reconcilersInfo := r.newCommonInfo(clh, ns, &loki) if err := r.checkFinalizer(ctx, desired); err != nil { diff --git a/internal/controller/flp/flp_controller.go b/internal/controller/flp/flp_controller.go index 1f2e82538..b45dc01ab 100644 --- a/internal/controller/flp/flp_controller.go +++ b/internal/controller/flp/flp_controller.go @@ -117,7 +117,7 @@ func (r *Reconciler) reconcile(ctx context.Context, clh *helper.Client, fc *flow ns := fc.Spec.GetNamespace() r.currentNamespace = ns previousNamespace := r.status.GetDeployedNamespace(fc) - loki := helper.NewLokiConfig(&fc.Spec.Loki, ns) + loki := helper.NewLokiConfig(&fc.Spec.Loki, ns, fc.Spec.Processor.HasExperimentalLokiGRPCClientProtocol()) cmn := r.newCommonInfo(clh, ns, &loki) r.watcher.Reset(ns) diff --git a/internal/controller/flp/flp_pipeline_builder.go b/internal/controller/flp/flp_pipeline_builder.go index 064af47ac..808496c5c 100644 --- a/internal/controller/flp/flp_pipeline_builder.go +++ b/internal/controller/flp/flp_pipeline_builder.go @@ -25,6 +25,9 @@ import ( const ( ovnkSecondary = "ovn-kubernetes" + + http = "http" + grpc = "grpc" ) type PipelineBuilder struct { @@ -262,7 +265,7 @@ func (b *PipelineBuilder) AddProcessorStages() error { if len(filters) > 0 { lokiStage = lokiStage.TransformFilter("filters-loki", newTransformFilter(filters)) } - + lokiClientProtocol := getLokiClientProtocol(&b.desired.Processor) lokiWrite := api.WriteLoki{ Labels: lokiLabels, BatchSize: int(b.desired.Loki.WriteBatchSize), @@ -276,14 +279,21 @@ func (b *PipelineBuilder) AddProcessorStages() error { TimestampLabel: "TimeFlowEndMs", TimestampScale: "1ms", TenantID: b.loki.TenantID, + ClientProtocol: lokiClientProtocol, + } + + // Configure gRPC-specific settings if using gRPC + if lokiWrite.ClientProtocol == grpc { + lokiWrite.GRPCConfig = b.buildGRPCConfig() } for k, v := range advancedConfig.StaticLabels { lokiWrite.StaticLabels[model.LabelName(k)] = model.LabelValue(v) } + // Configure client settings (TLS is shared between HTTP and gRPC) var authorization *promConfig.Authorization - if b.loki.UseHostToken() || b.loki.UseForwardToken() { + if lokiWrite.ClientProtocol == http && (b.loki.UseHostToken() || b.loki.UseForwardToken()) { b.volumes.AddToken(constants.FLPName) authorization = &promConfig.Authorization{ Type: "Bearer", @@ -291,27 +301,13 @@ func (b *PipelineBuilder) AddProcessorStages() error { } } - if b.loki.TLS.Enable { - if b.loki.TLS.InsecureSkipVerify { - lokiWrite.ClientConfig = &promConfig.HTTPClientConfig{ - Authorization: authorization, - TLSConfig: promConfig.TLSConfig{ - InsecureSkipVerify: true, - }, - } - } else { - caPath := b.volumes.AddCACertificate(&b.loki.TLS, "loki-certs") - lokiWrite.ClientConfig = &promConfig.HTTPClientConfig{ - Authorization: authorization, - TLSConfig: promConfig.TLSConfig{ - CAFile: caPath, - }, - } - } - } else { - lokiWrite.ClientConfig = &promConfig.HTTPClientConfig{ - Authorization: authorization, - } + // Build TLS config (different cert paths for HTTP vs gRPC) + useGRPCCerts := lokiWrite.ClientProtocol == grpc + tlsConfig := b.buildTLSConfig(&b.loki.TLS, b.loki.IngesterURL, useGRPCCerts) + + lokiWrite.ClientConfig = &promConfig.HTTPClientConfig{ + Authorization: authorization, + TLSConfig: tlsConfig, } lokiStage.WriteLoki("loki", lokiWrite) } @@ -768,3 +764,62 @@ func subnetLabelsToFLP(labels []flowslatest.SubnetLabel) []api.NetworkTransformS } return cats } + +// getLokiClientProtocol returns the client type, defaulting to http if experimental support is not configured +func getLokiClientProtocol(processorSpec *flowslatest.FlowCollectorFLP) string { + if processorSpec.HasExperimentalLokiGRPCClientProtocol() { + return grpc + } + return http +} + +// buildGRPCConfig builds the gRPC configuration for Loki writer +func (b *PipelineBuilder) buildGRPCConfig() *api.GRPCLokiConfig { + // TO-DO: Fill values from env vars + config := &api.GRPCLokiConfig{} + + if config.KeepAlive == "" { + config.KeepAlive = "30s" + } + if config.KeepAliveTimeout == "" { + config.KeepAliveTimeout = "5s" + } + + return config +} + +// buildTLSConfig builds TLS configuration for Loki client +// useGRPCCerts: true for gRPC (mutual TLS with client certs), false for HTTP (CA only) +func (b *PipelineBuilder) buildTLSConfig(tlsConfig *flowslatest.ClientTLS, serverAddress string, useGRPCCerts bool) promConfig.TLSConfig { + if !tlsConfig.Enable { + return promConfig.TLSConfig{} + } + + config := promConfig.TLSConfig{ + InsecureSkipVerify: tlsConfig.InsecureSkipVerify, + } + + if !tlsConfig.InsecureSkipVerify { + if useGRPCCerts { + // gRPC uses mutual TLS with client certificates + caPath, certPath, keyPath := b.volumes.AddMutualTLSCertificates(tlsConfig, "loki-grpc-certs") + config.CAFile = caPath + config.CertFile = certPath + config.KeyFile = keyPath + + // Set ServerName for certificate verification + // Extract hostname from server address (remove port) + serverName := serverAddress + if idx := strings.LastIndex(serverName, ":"); idx != -1 { + serverName = serverName[:idx] + } + config.ServerName = serverName + } else { + // HTTP uses only CA certificate + caPath := b.volumes.AddCACertificate(tlsConfig, "loki-certs") + config.CAFile = caPath + } + } + + return config +} diff --git a/internal/controller/flp/flp_test.go b/internal/controller/flp/flp_test.go index ff9e2acc4..46f86a735 100644 --- a/internal/controller/flp/flp_test.go +++ b/internal/controller/flp/flp_test.go @@ -174,14 +174,14 @@ func monoBuilder(ns string, cfg *flowslatest.FlowCollectorSpec) monolithBuilder } func monoBuilderWithMetrics(ns string, cfg *flowslatest.FlowCollectorSpec, metrics *metricslatest.FlowMetricList) monolithBuilder { - loki := helper.NewLokiConfig(&cfg.Loki, "any") + loki := helper.NewLokiConfig(&cfg.Loki, "any", false) info := reconcilers.Common{Namespace: ns, Loki: &loki, ClusterInfo: &cluster.Info{}} b, _ := newMonolithBuilder(info.NewInstance(image, status.Instance{}), cfg, metrics, nil) return b } func transfBuilder(ns string, cfg *flowslatest.FlowCollectorSpec) transfoBuilder { - loki := helper.NewLokiConfig(&cfg.Loki, "any") + loki := helper.NewLokiConfig(&cfg.Loki, "any", false) info := reconcilers.Common{Namespace: ns, Loki: &loki, ClusterInfo: &cluster.Info{}} b, _ := newTransfoBuilder(info.NewInstance(image, status.Instance{}), cfg, &metricslatest.FlowMetricList{}, nil) return b diff --git a/internal/controller/flp/metrics_api_test.go b/internal/controller/flp/metrics_api_test.go index 5777c9d4b..e49b4ecde 100644 --- a/internal/controller/flp/metrics_api_test.go +++ b/internal/controller/flp/metrics_api_test.go @@ -34,7 +34,7 @@ func getConfiguredMetrics(cm *corev1.ConfigMap) (api.MetricsItems, error) { func defaultBuilderWithMetrics(metrics *metricslatest.FlowMetricList) (monolithBuilder, error) { cfg := getConfig() - loki := helper.NewLokiConfig(&cfg.Loki, "any") + loki := helper.NewLokiConfig(&cfg.Loki, "any", false) info := reconcilers.Common{Namespace: "namespace", Loki: &loki, ClusterInfo: &cluster.Info{}} return newMonolithBuilder(info.NewInstance(image, status.Instance{}), &cfg, metrics, nil) } diff --git a/internal/pkg/helper/loki_config.go b/internal/pkg/helper/loki_config.go index 916bfc6e3..6150fb9e6 100644 --- a/internal/pkg/helper/loki_config.go +++ b/internal/pkg/helper/loki_config.go @@ -10,7 +10,7 @@ type LokiConfig struct { flowslatest.LokiManualParams } -func NewLokiConfig(spec *flowslatest.FlowCollectorLoki, namespace string) LokiConfig { +func NewLokiConfig(spec *flowslatest.FlowCollectorLoki, namespace string, useGRPC bool) LokiConfig { loki := LokiConfig{} switch spec.Mode { case flowslatest.LokiModeLokiStack: @@ -19,21 +19,48 @@ func NewLokiConfig(spec *flowslatest.FlowCollectorLoki, namespace string) LokiCo ns = spec.LokiStack.Namespace } gatewayURL := fmt.Sprintf("https://%s-gateway-http.%s.svc:8080/api/logs/v1/network/", spec.LokiStack.Name, ns) + ingesterURL := gatewayURL + if useGRPC { + ingesterURL = fmt.Sprintf("%s-distributor-grpc.%s.svc:9095", spec.LokiStack.Name, ns) + } + // Configure TLS based on client type + tlsConfig := flowslatest.ClientTLS{ + Enable: true, + } + + // Set TLS certificates based on the connection type + if useGRPC { + // For gRPC ingester connections: use the Loki signing CA + tlsConfig.CACert = flowslatest.CertificateReference{ + Type: flowslatest.RefTypeConfigMap, + Name: fmt.Sprintf("%s-ca-bundle", spec.LokiStack.Name), + Namespace: spec.LokiStack.Namespace, + CertFile: "service-ca.crt", + } + tlsConfig.UserCert = flowslatest.CertificateReference{ + Type: flowslatest.RefTypeSecret, + Name: fmt.Sprintf("%s-distributor-grpc", spec.LokiStack.Name), + Namespace: spec.LokiStack.Namespace, + CertFile: "tls.crt", + CertKey: "tls.key", + } + } else { + // For HTTP gateway connections: use the OpenShift service serving CA + tlsConfig.CACert = flowslatest.CertificateReference{ + Type: flowslatest.RefTypeConfigMap, + Name: fmt.Sprintf("%s-gateway-ca-bundle", spec.LokiStack.Name), + Namespace: spec.LokiStack.Namespace, + CertFile: "service-ca.crt", + } + } + loki.LokiManualParams = flowslatest.LokiManualParams{ QuerierURL: gatewayURL, - IngesterURL: gatewayURL, + IngesterURL: ingesterURL, StatusURL: fmt.Sprintf("https://%s-query-frontend-http.%s.svc:3100/", spec.LokiStack.Name, ns), TenantID: "network", AuthToken: flowslatest.LokiAuthForwardUserToken, - TLS: flowslatest.ClientTLS{ - Enable: true, - CACert: flowslatest.CertificateReference{ - Type: flowslatest.RefTypeConfigMap, - Name: fmt.Sprintf("%s-gateway-ca-bundle", spec.LokiStack.Name), - Namespace: spec.LokiStack.Namespace, - CertFile: "service-ca.crt", - }, - }, + TLS: tlsConfig, StatusTLS: flowslatest.ClientTLS{ Enable: true, CACert: flowslatest.CertificateReference{ diff --git a/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go b/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go index e3016ed73..4c04b6361 100644 --- a/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go +++ b/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go @@ -49,7 +49,6 @@ type PromConnectionInfo struct { type MetricsItem struct { Name string `yaml:"name" json:"name" doc:"the metric name"` Type MetricEncodeOperationEnum `yaml:"type" json:"type" doc:"(enum) one of the following:"` - Help string `yaml:"help,omitempty" json:"help,omitempty" doc:"the metric help text"` Filters []MetricsFilter `yaml:"filters" json:"filters" doc:"a list of criteria to filter entries by"` ValueKey string `yaml:"valueKey" json:"valueKey" doc:"entry key from which to resolve metric value"` Labels []string `yaml:"labels" json:"labels" doc:"labels to be associated with the metric"` diff --git a/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/write_loki.go b/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/write_loki.go index cdc9a36bc..8225ad276 100644 --- a/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/write_loki.go +++ b/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/write_loki.go @@ -20,6 +20,7 @@ package api import ( "errors" "fmt" + "time" promConfig "github.com/prometheus/common/config" "github.com/prometheus/common/model" @@ -46,6 +47,15 @@ type WriteLoki struct { TimestampScale string `yaml:"timestampScale,omitempty" json:"timestampScale,omitempty" doc:"timestamp units scale (e.g. for UNIX = 1s)"` Format string `yaml:"format,omitempty" json:"format,omitempty" doc:"the format of each line: printf (writes using golang's default map printing), fields (writes one key and value field per line) or json (default)"` Reorder bool `yaml:"reorder,omitempty" json:"reorder,omitempty" doc:"reorder json map keys"` + + // Client protocol selection + ClientProtocol string `yaml:"clientProtocol,omitempty" json:"clientProtocol,omitempty" doc:"type of client protocol to use: 'http' or 'grpc' (default: 'http')"` + GRPCConfig *GRPCLokiConfig `yaml:"grpcConfig,omitempty" json:"grpcConfig,omitempty" doc:"gRPC client configuration (used only for gRPC client type)"` +} + +type GRPCLokiConfig struct { + KeepAlive string `yaml:"keepAlive,omitempty" json:"keepAlive,omitempty" doc:"keep alive interval"` + KeepAliveTimeout string `yaml:"keepAliveTimeout,omitempty" json:"keepAliveTimeout,omitempty" doc:"keep alive timeout"` } func (w *WriteLoki) SetDefaults() { @@ -76,6 +86,23 @@ func (w *WriteLoki) SetDefaults() { if w.Format == "" { w.Format = "json" } + if w.ClientProtocol == "" { + w.ClientProtocol = "http" + } + + // Set defaults for gRPC config if gRPC client protocol is selected + if w.ClientProtocol == "grpc" && w.GRPCConfig != nil { + w.GRPCConfig.SetDefaults() + } +} + +func (g *GRPCLokiConfig) SetDefaults() { + if g.KeepAlive == "" { + g.KeepAlive = "30s" + } + if g.KeepAliveTimeout == "" { + g.KeepAliveTimeout = "5s" + } } func (w *WriteLoki) Validate() error { @@ -85,11 +112,51 @@ func (w *WriteLoki) Validate() error { if w.TimestampScale == "" { return errors.New("timestampUnit must be a valid Duration > 0 (e.g. 1m, 1s or 1ms)") } - if w.URL == "" { - return errors.New("url can't be empty") - } if w.BatchSize <= 0 { return fmt.Errorf("invalid batchSize: %v. Required > 0", w.BatchSize) } + + // Validate client protocol + if w.ClientProtocol != "" && w.ClientProtocol != "http" && w.ClientProtocol != "grpc" { + return fmt.Errorf("invalid clientProtocol: %s. Must be 'http' or 'grpc'", w.ClientProtocol) + } + + // Validate based on client protocol + switch w.ClientProtocol { + case "http", "": + if w.URL == "" { + return errors.New("url can't be empty for HTTP client") + } + case "grpc": + if w.URL == "" { + return errors.New("url can't be empty for gRPC client") + } + if w.GRPCConfig == nil { + return errors.New("grpcConfig is required when using gRPC client protocol") + } + if err := w.GRPCConfig.Validate(); err != nil { + return fmt.Errorf("gRPC config validation failed: %w", err) + } + } + + return nil +} + +func (g *GRPCLokiConfig) Validate() error { + if g == nil { + return errors.New("gRPC config cannot be nil") + } + // Validate duration fields + if g.KeepAlive != "" { + if _, err := time.ParseDuration(g.KeepAlive); err != nil { + return fmt.Errorf("invalid keepAlive duration: %w", err) + } + } + if g.KeepAliveTimeout != "" { + if _, err := time.ParseDuration(g.KeepAliveTimeout); err != nil { + return fmt.Errorf("invalid keepAliveTimeout duration: %w", err) + } + } + return nil } diff --git a/vendor/modules.txt b/vendor/modules.txt index e04c7f638..ba39b8d90 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -161,7 +161,11 @@ github.com/munnerz/goautoneg # github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f ## explicit github.com/mwitkow/go-conntrack +<<<<<<< HEAD # github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99 +======= +# github.com/netobserv/flowlogs-pipeline v1.9.2-community => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc +>>>>>>> 6b4e3db1 (Support for configure grpc client to send logs to Loki) ## explicit; go 1.24.0 github.com/netobserv/flowlogs-pipeline/pkg/api github.com/netobserv/flowlogs-pipeline/pkg/config @@ -1258,3 +1262,4 @@ sigs.k8s.io/structured-merge-diff/v6/value # sigs.k8s.io/yaml v1.6.0 ## explicit; go 1.22 sigs.k8s.io/yaml +# github.com/netobserv/flowlogs-pipeline => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc From a828c256879766fa5b78910d3f6b681d085f681f Mon Sep 17 00:00:00 2001 From: Leandro Beretta Date: Thu, 9 Oct 2025 13:38:01 -0300 Subject: [PATCH 2/3] fix vendors --- go.sum | 2 -- vendor/modules.txt | 6 +----- 2 files changed, 1 insertion(+), 7 deletions(-) diff --git a/go.sum b/go.sum index ff29b641d..06efca503 100644 --- a/go.sum +++ b/go.sum @@ -119,8 +119,6 @@ github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f h1:KUppIJq7/+SVif2QVs3tOP0zanoHgBEVAwHxUSIzRqU= github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= -github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99 h1:amQHvOeTXw04c5PAEJ4yvvIpFrcuBV7qeJsgfRxq4NA= -github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99/go.mod h1:RCD0EdA1NFZ1xXhSePB8TRQahfNqYTWlHD2lKDfNBW0= github.com/netobserv/netobserv-ebpf-agent v1.9.2-community h1:ghW16OO4QRWj0Uh1gMYX+NjAlgx2sZmCsO3Tkwoj4Do= github.com/netobserv/netobserv-ebpf-agent v1.9.2-community/go.mod h1:17OaUNAwx0LxoeV/SaHlJIJP6bpN7zSvUP3GtZelESQ= github.com/netsampler/goflow2 v1.3.7 h1:XZaTy8kkMnGXpJ9hS3KbO1McyrFTpVNhVFEx9rNhMmc= diff --git a/vendor/modules.txt b/vendor/modules.txt index ba39b8d90..f1d9cc77d 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -161,11 +161,7 @@ github.com/munnerz/goautoneg # github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f ## explicit github.com/mwitkow/go-conntrack -<<<<<<< HEAD -# github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99 -======= -# github.com/netobserv/flowlogs-pipeline v1.9.2-community => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc ->>>>>>> 6b4e3db1 (Support for configure grpc client to send logs to Loki) +# github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99 => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc ## explicit; go 1.24.0 github.com/netobserv/flowlogs-pipeline/pkg/api github.com/netobserv/flowlogs-pipeline/pkg/config From 83348996b548448253125f6c69caf234c0d96328 Mon Sep 17 00:00:00 2001 From: Leandro Beretta Date: Thu, 9 Oct 2025 13:46:00 -0300 Subject: [PATCH 3/3] fix flp dep --- go.mod | 2 +- go.sum | 24 +++++++++---------- .../flowlogs-pipeline/pkg/api/encode_prom.go | 1 + vendor/modules.txt | 4 ++-- 4 files changed, 16 insertions(+), 15 deletions(-) diff --git a/go.mod b/go.mod index 76db49222..1fd42d054 100644 --- a/go.mod +++ b/go.mod @@ -120,4 +120,4 @@ require ( sigs.k8s.io/structured-merge-diff/v6 v6.3.0 // indirect ) -replace github.com/netobserv/flowlogs-pipeline => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc +replace github.com/netobserv/flowlogs-pipeline => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009164306-d01e3863647a diff --git a/go.sum b/go.sum index 06efca503..24cb7f469 100644 --- a/go.sum +++ b/go.sum @@ -101,8 +101,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= -github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc h1:0O0vz3oHT1pj2i3dDTeIlW1LrqrmbGKw711WAjpjBoM= -github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc/go.mod h1:JKatcqNUQeNfSGMpQ73a6tsbmw/ztOZunqOYfNJ7Opg= +github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009164306-d01e3863647a h1:pS4iRF2fco9qLbTvl/s+tLBIHMErMbin6tYXZWcK6S4= +github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009164306-d01e3863647a/go.mod h1:nHLh9A9s0lsWGQqbWpooUlB/Y/H2ik1knLjncY1L7BY= github.com/mailru/easyjson v0.9.0 h1:PrnmzHw7262yW8sTBwxi1PdJA3Iw/EKBa8psRf7d9a4= github.com/mailru/easyjson v0.9.0/go.mod h1:1+xMtQp2MRNVL/V1bOzuP3aP8VNwRW55fQUto+XFtTU= github.com/maruel/natural v1.1.1 h1:Hja7XhhmvEFhcByqDoHz9QZbkWey+COd9xWfCfn1ioo= @@ -151,23 +151,23 @@ github.com/prometheus/procfs v0.17.0/go.mod h1:oPQLaDAMRbA+u8H5Pbfq+dl3VDAvHxMUO github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR38lUII= github.com/rogpeppe/go-internal v1.13.1/go.mod h1:uMEvuHeurkdAXX61udpOXGD/AzZDWNMNyH2VO9fmH0o= github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= -github.com/sagikazarmark/locafero v0.9.0 h1:GbgQGNtTrEmddYDSAH9QLRyfAHY12md+8YFTqyMTC9k= -github.com/sagikazarmark/locafero v0.9.0/go.mod h1:UBUyz37V+EdMS3hDF3QWIiVr/2dPrx49OMO0Bn0hJqk= +github.com/sagikazarmark/locafero v0.11.0 h1:1iurJgmM9G3PA/I+wWYIOw/5SyBtxapeHDcg+AAIFXc= +github.com/sagikazarmark/locafero v0.11.0/go.mod h1:nVIGvgyzw595SUSUE6tvCp3YYTeHs15MvlmU87WwIik= github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= -github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo= -github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0= -github.com/spf13/afero v1.14.0 h1:9tH6MapGnn/j0eb0yIXiLjERO8RB6xIVZRDCX7PtqWA= -github.com/spf13/afero v1.14.0/go.mod h1:acJQ8t0ohCGuMN3O+Pv0V0hgMxNYDlvdk+VTfyZmbYo= -github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y= -github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= +github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 h1:+jumHNA0Wrelhe64i8F6HNlS8pkoyMv5sreGx2Ry5Rw= +github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8/go.mod h1:3n1Cwaq1E1/1lhQhtRK2ts/ZwZEhjcQeJQ1RuC6Q/8U= +github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I= +github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg= +github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY= +github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo= github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo= github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0= github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk= github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= -github.com/spf13/viper v1.20.1 h1:ZMi+z/lvLyPSCoNtFCpqjy0S4kPbirhpTMwl8BkW9X4= -github.com/spf13/viper v1.20.1/go.mod h1:P9Mdzt1zoHIG8m2eZQinpiBjo6kCmZSKBClNNqjJvu4= +github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU= +github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY= github.com/stoewer/go-strcase v1.3.1 h1:iS0MdW+kVTxgMoE1LAZyMiYJFKlOzLooE4MxjirtkAs= github.com/stoewer/go-strcase v1.3.1/go.mod h1:fAH5hQ5pehh+j3nZfvwdk2RgEgQjAoM8wodgtPmh1xo= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= diff --git a/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go b/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go index 4c04b6361..e3016ed73 100644 --- a/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go +++ b/vendor/github.com/netobserv/flowlogs-pipeline/pkg/api/encode_prom.go @@ -49,6 +49,7 @@ type PromConnectionInfo struct { type MetricsItem struct { Name string `yaml:"name" json:"name" doc:"the metric name"` Type MetricEncodeOperationEnum `yaml:"type" json:"type" doc:"(enum) one of the following:"` + Help string `yaml:"help,omitempty" json:"help,omitempty" doc:"the metric help text"` Filters []MetricsFilter `yaml:"filters" json:"filters" doc:"a list of criteria to filter entries by"` ValueKey string `yaml:"valueKey" json:"valueKey" doc:"entry key from which to resolve metric value"` Labels []string `yaml:"labels" json:"labels" doc:"labels to be associated with the metric"` diff --git a/vendor/modules.txt b/vendor/modules.txt index f1d9cc77d..0236c8782 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -161,7 +161,7 @@ github.com/munnerz/goautoneg # github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f ## explicit github.com/mwitkow/go-conntrack -# github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99 => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc +# github.com/netobserv/flowlogs-pipeline v1.9.2-community.0.20251006073644-7e3fe86c1e99 => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009164306-d01e3863647a ## explicit; go 1.24.0 github.com/netobserv/flowlogs-pipeline/pkg/api github.com/netobserv/flowlogs-pipeline/pkg/config @@ -1258,4 +1258,4 @@ sigs.k8s.io/structured-merge-diff/v6/value # sigs.k8s.io/yaml v1.6.0 ## explicit; go 1.22 sigs.k8s.io/yaml -# github.com/netobserv/flowlogs-pipeline => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009135050-3d8583a3cfbc +# github.com/netobserv/flowlogs-pipeline => github.com/leandroberetta/flowlogs-pipeline v0.0.0-20251009164306-d01e3863647a