Sample configuration for AWS EC2 handling 25k entities and 10k metrics/sec (small) with NGINX Ingress controller

Download this sample AWS EC2 handling 25k entities and 10k metrics/sec (small) configuration provided by ITRS.

# Example Obcerv configuration for AWS EC2 handling 25k Obcerv entities and 10k metrics/sec, and 10k OpenTelemetry
# spans/sec (pre-sampling).
#
# Nodes: (4) c5.4xlarge (16CPU, 32GB)
#
# The resource requests total ~33 cores and ~79GiB memory (assuming collection-agent DaemonSet runs on 4 nodes)
# and includes Linkerd resources.
#
# Disk requirements:
# - Timescale:
#   - 4 x 512 GiB timeseries data disk for each replica (x2)
#   - 50 GiB data disk for each replica (x2)
#   - 50 GiB WAL disk for each replica (x2)
# - Kafka broker: 100 GiB for each replica (x3)
# - Kafka controller: 1 GiB for each replica (x1)
# - Loki: 30 GiB
# - etcd: 1 GiB for each replica (x3)
# - Downsampled Metrics:
#   - Raw: 5 GiB for each replica (x1)
#   - Bucketed: 5 GiB for each replica (x1)
#
# The configuration references a default storage class named `gp3` which uses EBS gp3 volumes. This storage class should
# be configured with the default minimum gp3 settings of 3000 IOPS and 125 MiB/s throughput - you can create
# this class or change the config to use a class of your own, but it should be similar in performance.
#
# This configuration is based upon a certain number of Obcerv entities, average metrics per entity, and
# average metrics collection interval. The following function can be used to figure out what type of load to expect:
#
# metrics/sec = (Obcerv entities * metrics/entity) / average metrics collection interval
#
# In this example configuration, we have the following:
#
# 10,000 metrics/sec = (25,000 Obcerv entities * 4 metrics/entity) / 10 seconds average metrics collection interval
#
# NOTE: Ingestion, storage, and retrieval of OpenTelemetry spans is a beta feature.
#
# Additionally, the configuration is based upon a certain number of OpenTelemetry spans per second that are sampled
# based upon the following rules:
# - Error traces are always sampled
# - Target sampling probability per endpoint (corresponds to the name of the root span) is 0.01
# - Target sampling rate / second / endpoint (corresponds to the name of the root span) is 0.5
# - Root span duration outlier quantile is 0.95. The durations of all root spans are tracked and used to make guesses about
#   abnormally long spans
#

defaultStorageClass: "gp3"
apps:
  externalHostname: "obcerv.mydomain.internal"
  ingress:
    annotations:
      kubernetes.io/ingress.class: "nginx"
      nginx.org/mergeable-ingress-type: "master"
ingestion:
  externalHostname: "obcerv-ingestion.mydomain.internal"
  replicas: 2
  ingress:
    annotations:
      kubernetes.io/ingress.class: "nginx"
      nginx.ingress.kubernetes.io/backend-protocol: "GRPC"
  resources:
    requests:
      memory: "512Mi"
      cpu: "500m"
    limits:
      memory: "512Mi"
      cpu: "750m"
  # NOTE: OpenTelemetry Traces ingestion is a beta feature and resources may need to be adjusted based on ingestion rate.
  traces:
    jvmOpts: "-Xms1G -Xmx1536M"
    resources:
      requests:
        memory: "1536Mi"
        cpu: "750m"
      limits:
        memory: "2Gi"
        cpu: "1"
iam:
  ingress:
    annotations:
      kubernetes.io/ingress.class: "nginx"
      nginx.org/mergeable-ingress-type: "minion"
kafka:
  replicas: 3
  diskSize: "100Gi"
  resources:
    requests:
      memory: "3Gi"
      cpu: "1"
    limits:
      memory: "3Gi"
      cpu: "2"
timescale:
  sharedBuffersPercentage: 40
  clusterSize: 2
  dataDiskSize: "50Gi"
  timeseriesDiskCount: 4
  timeseriesDiskSize: "512Gi"
  walDiskSize: "50Gi"
  resources:
    requests:
      memory: "14Gi"
      cpu: "6"
    limits:
      memory: "14Gi"
      cpu: "6"
  retention:
    entity_attributes:
      chunkSize: 2d
      retention: 1y
    metrics:
      chunkSize: 8h
      retention: 30d
    metrics_5m:
      chunkSize: 1d
      retention: 90d
    metrics_1h:
      chunkSize: 5d
      retention: 180d
    metrics_1d:
      chunkSize: 20d
      retention: 1y
    statuses:
      chunkSize: 7d
      retention: 1y
    signal_details:
      chunkSize: 1d
      retention: 30d
    traces:
      chunkSize: 4h
      retention: 5d
    # The retention for the traces table applies to this table.
    span_links:
      chunkSize: 1d
loki:
  diskSize: "30Gi"
sinkd:
  timeseriesCacheMaxSize: 1000000
  jvmOpts: "-Xms512M -Xmx1536M -XX:MaxDirectMemorySize=100M"
  rawJvmOpts: "-Xms1536M -Xmx1536M"
  replicas: 2
  rawReplicas: 2
  resources:
    requests:
      memory: "1280Mi"
      cpu: "500m"
    limits:
      memory: "2Gi"
      cpu: "2"
  rawResources:
    requests:
      memory: "1Gi"
      cpu: "250m"
    limits:
      memory: "2Gi"
      cpu: "2"
  metrics:
    consumerProperties:
      max.partition.fetch.bytes: 524288
  dsMetrics:
    consumerProperties:
      max.partition.fetch.bytes: 1048576
  loki:
    consumerProperties:
      max.partition.fetch.bytes: 1048576
  entities:
    consumerProperties:
      max.partition.fetch.bytes: 1048576
  signals:
    consumerProperties:
      max.partition.fetch.bytes: 1048576
platformd:
  replicas: 2
  resources:
    requests:
      memory: "1536Mi"
      cpu: "1"
    limits:
      memory: "2Gi"
      cpu: "2250m"
dpd:
  jvmOpts: "-Xmx3G"
  maxEntitySerdeCacheEntries: 25000
  consumerProperties:
    fetch.min.bytes: 524288
  metricsMultiplexer:
    maxFilterResultCacheSize: 500000
    maxConcurrentOps: 100
    localParallelism: 6
  selfMonitoringThresholds:
    metrics_partition_lag_warn: 100000
    metrics_partition_lag_critical: 500000
  resources:
    requests:
      memory: "4Gi"
      cpu: "1"
    limits:
      memory: "5Gi"
      cpu: "2"
downsampledMetricsStream:
  consumerProperties:
    fetch.min.bytes: 524288
    max.partition.fetch.bytes: 1048576
  resources:
    requests:
      memory: "3Gi"
      cpu: "1"
    limits:
      memory: "3Gi"
      cpu: "2700m"
  bucketedConsumerProperties:
    fetch.min.bytes: 524288
    max.partition.fetch.bytes: 1048576
  bucketedJvmOpts: "-XX:InitialRAMPercentage=75 -XX:MaxRAMPercentage=75"
  bucketedResources:
    requests:
      memory: "3Gi"
      cpu: "1"
    limits:
      memory: "6Gi"
      cpu: "4"
  rocksdb:
    raw:
      indexAndFilterRatio: 0.5
      memoryMib: 100
      writeBufferMib: 8
      writeBufferRatio: 0.25
    bucketed:
      indexAndFilterRatio: 0.5
      memoryMib: 100
      writeBufferMib: 8
      writeBufferRatio: 0.25
entityStream:
  intermediate:
    consumerProperties:
      max.partition.fetch.bytes: 1048576
    storedEntitiesCacheSize: 1000
    resources:
      requests:
        memory: "1Gi"
        cpu: "400m"
      limits:
        memory: "2Gi"
        cpu: "1125m"
  final:
    consumerProperties:
      max.partition.fetch.bytes: 1048576
    resources:
      requests:
        memory: "1350Mi"
        cpu: "300m"
      limits:
        memory: "2Gi"
        cpu: "3"
signalsStream:
  consumerProperties:
    max.partition.fetch.bytes: 1048576
  resources:
    requests:
      memory: "768Mi"
      cpu: "150m"
    limits:
      memory: "1536Mi"
      cpu: "1200m"
etcd:
  replicas: 3
["Obcerv"] ["User Guide", "Technical Reference"]

Was this topic helpful?