Usage Recommendations

CPU Scaling Governor

Always use the performance scaling governor. The on-demand scaling governor works much worse with constantly high demand.

  1. $ echo 'performance' | sudo tee /sys/devices/system/cpu/cpu*/cpufreq/scaling_governor

CPU Limitations

Processors can overheat. Use dmesg to see if the CPU’s clock rate was limited due to overheating.
The restriction can also be set externally at the datacenter level. You can use turbostat to monitor it under a load.

RAM

For small amounts of data (up to ~200 GB compressed), it is best to use as much memory as the volume of data.
For large amounts of data and when processing interactive (online) queries, you should use a reasonable amount of RAM (128 GB or more) so the hot data subset will fit in the cache of pages.
Even for data volumes of ~50 TB per server, using 128 GB of RAM significantly improves query performance compared to 64 GB.

Do not disable overcommit. The value cat /proc/sys/vm/overcommit_memory should be 0 or 1. Run

  1. $ echo 0 | sudo tee /proc/sys/vm/overcommit_memory

Huge Pages

Always disable transparent huge pages. It interferes with memory allocators, which leads to significant performance degradation.

  1. $ echo 'never' | sudo tee /sys/kernel/mm/transparent_hugepage/enabled

Use perf top to watch the time spent in the kernel for memory management.
Permanent huge pages also do not need to be allocated.

Storage Subsystem

If your budget allows you to use SSD, use SSD.
If not, use HDD. SATA HDDs 7200 RPM will do.

Give preference to a lot of servers with local hard drives over a smaller number of servers with attached disk shelves.
But for storing archives with rare queries, shelves will work.

RAID

When using HDD, you can combine their RAID-10, RAID-5, RAID-6 or RAID-50.
For Linux, software RAID is better (with mdadm). We don’t recommend using LVM.
When creating RAID-10, select the far layout.
If your budget allows, choose RAID-10.

If you have more than 4 disks, use RAID-6 (preferred) or RAID-50, instead of RAID-5.
When using RAID-5, RAID-6 or RAID-50, always increase stripe_cache_size, since the default value is usually not the best choice.

  1. $ echo 4096 | sudo tee /sys/block/md2/md/stripe_cache_size

Calculate the exact number from the number of devices and the block size, using the formula: 2 * num_devices * chunk_size_in_bytes / 4096.

A block size of 1024 KB is sufficient for all RAID configurations.
Never set the block size too small or too large.

You can use RAID-0 on SSD.
Regardless of RAID use, always use replication for data security.

Enable NCQ with a long queue. For HDD, choose the CFQ scheduler, and for SSD, choose noop. Don’t reduce the ‘readahead’ setting.
For HDD, enable the write cache.

File System

Ext4 is the most reliable option. Set the mount options noatime, nobarrier.
XFS is also suitable, but it hasn’t been as thoroughly tested with ClickHouse.
Most other file systems should also work fine. File systems with delayed allocation work better.

Linux Kernel

Don’t use an outdated Linux kernel.

Network

If you are using IPv6, increase the size of the route cache.
The Linux kernel prior to 3.2 had a multitude of problems with IPv6 implementation.

Use at least a 10 GB network, if possible. 1 Gb will also work, but it will be much worse for patching replicas with tens of terabytes of data, or for processing distributed queries with a large amount of intermediate data.

ZooKeeper

You are probably already using ZooKeeper for other purposes. You can use the same installation of ZooKeeper, if it isn’t already overloaded.

It’s best to use a fresh version of ZooKeeper – 3.4.9 or later. The version in stable Linux distributions may be outdated.

You should never use manually written scripts to transfer data between different ZooKeeper clusters, because the result will be incorrect for sequential nodes. Never use the “zkcopy” utility for the same reason: https://github.com/ksprojects/zkcopy/issues/15

If you want to divide an existing ZooKeeper cluster into two, the correct way is to increase the number of its replicas and then reconfigure it as two independent clusters.

Do not run ZooKeeper on the same servers as ClickHouse. Because ZooKeeper is very sensitive for latency and ClickHouse may utilize all available system resources.

With the default settings, ZooKeeper is a time bomb:

The ZooKeeper server won’t delete files from old snapshots and logs when using the default configuration (see autopurge), and this is the responsibility of the operator.

This bomb must be defused.

The ZooKeeper (3.5.1) configuration below is used in the Yandex.Metrica production environment as of May 20, 2017:

zoo.cfg:

  1. # http://hadoop.apache.org/zookeeper/docs/current/zookeeperAdmin.html
  2. # The number of milliseconds of each tick
  3. tickTime=2000
  4. # The number of ticks that the initial
  5. # synchronization phase can take
  6. initLimit=30000
  7. # The number of ticks that can pass between
  8. # sending a request and getting an acknowledgement
  9. syncLimit=10
  10. maxClientCnxns=2000
  11. maxSessionTimeout=60000000
  12. # the directory where the snapshot is stored.
  13. dataDir=/opt/zookeeper/{{ cluster['name'] }}/data
  14. # Place the dataLogDir to a separate physical disc for better performance
  15. dataLogDir=/opt/zookeeper/{{ cluster['name'] }}/logs
  16. autopurge.snapRetainCount=10
  17. autopurge.purgeInterval=1
  18. # To avoid seeks ZooKeeper allocates space in the transaction log file in
  19. # blocks of preAllocSize kilobytes. The default block size is 64M. One reason
  20. # for changing the size of the blocks is to reduce the block size if snapshots
  21. # are taken more often. (Also, see snapCount).
  22. preAllocSize=131072
  23. # Clients can submit requests faster than ZooKeeper can process them,
  24. # especially if there are a lot of clients. To prevent ZooKeeper from running
  25. # out of memory due to queued requests, ZooKeeper will throttle clients so that
  26. # there is no more than globalOutstandingLimit outstanding requests in the
  27. # system. The default limit is 1,000.ZooKeeper logs transactions to a
  28. # transaction log. After snapCount transactions are written to a log file a
  29. # snapshot is started and a new transaction log file is started. The default
  30. # snapCount is 10,000.
  31. snapCount=3000000
  32. # If this option is defined, requests will be will logged to a trace file named
  33. # traceFile.year.month.day.
  34. #traceFile=
  35. # Leader accepts client connections. Default value is "yes". The leader machine
  36. # coordinates updates. For higher update throughput at thes slight expense of
  37. # read throughput the leader can be configured to not accept clients and focus
  38. # on coordination.
  39. leaderServes=yes
  40. standaloneEnabled=false
  41. dynamicConfigFile=/etc/zookeeper-{{ cluster['name'] }}/conf/zoo.cfg.dynamic

Java version:

  1. Java(TM) SE Runtime Environment (build 1.8.0_25-b17)
  2. Java HotSpot(TM) 64-Bit Server VM (build 25.25-b02, mixed mode)

JVM parameters:

  1. NAME=zookeeper-{{ cluster['name'] }}
  2. ZOOCFGDIR=/etc/$NAME/conf
  3. # TODO this is really ugly
  4. # How to find out, which jars are needed?
  5. # seems, that log4j requires the log4j.properties file to be in the classpath
  6. CLASSPATH="$ZOOCFGDIR:/usr/build/classes:/usr/build/lib/*.jar:/usr/share/zookeeper/zookeeper-3.5.1-metrika.jar:/usr/share/zookeeper/slf4j-log4j12-1.7.5.jar:/usr/share/zookeeper/slf4j-api-1.7.5.jar:/usr/share/zookeeper/servlet-api-2.5-20081211.jar:/usr/share/zookeeper/netty-3.7.0.Final.jar:/usr/share/zookeeper/log4j-1.2.16.jar:/usr/share/zookeeper/jline-2.11.jar:/usr/share/zookeeper/jetty-util-6.1.26.jar:/usr/share/zookeeper/jetty-6.1.26.jar:/usr/share/zookeeper/javacc.jar:/usr/share/zookeeper/jackson-mapper-asl-1.9.11.jar:/usr/share/zookeeper/jackson-core-asl-1.9.11.jar:/usr/share/zookeeper/commons-cli-1.2.jar:/usr/src/java/lib/*.jar:/usr/etc/zookeeper"
  7. ZOOCFG="$ZOOCFGDIR/zoo.cfg"
  8. ZOO_LOG_DIR=/var/log/$NAME
  9. USER=zookeeper
  10. GROUP=zookeeper
  11. PIDDIR=/var/run/$NAME
  12. PIDFILE=$PIDDIR/$NAME.pid
  13. SCRIPTNAME=/etc/init.d/$NAME
  14. JAVA=/usr/bin/java
  15. ZOOMAIN="org.apache.zookeeper.server.quorum.QuorumPeerMain"
  16. ZOO_LOG4J_PROP="INFO,ROLLINGFILE"
  17. JMXLOCALONLY=false
  18. JAVA_OPTS="-Xms{{ cluster.get('xms','128M') }} \
  19. -Xmx{{ cluster.get('xmx','1G') }} \
  20. -Xloggc:/var/log/$NAME/zookeeper-gc.log \
  21. -XX:+UseGCLogFileRotation \
  22. -XX:NumberOfGCLogFiles=16 \
  23. -XX:GCLogFileSize=16M \
  24. -verbose:gc \
  25. -XX:+PrintGCTimeStamps \
  26. -XX:+PrintGCDateStamps \
  27. -XX:+PrintGCDetails
  28. -XX:+PrintTenuringDistribution \
  29. -XX:+PrintGCApplicationStoppedTime \
  30. -XX:+PrintGCApplicationConcurrentTime \
  31. -XX:+PrintSafepointStatistics \
  32. -XX:+UseParNewGC \
  33. -XX:+UseConcMarkSweepGC \
  34. -XX:+CMSParallelRemarkEnabled"

Salt init:

  1. description "zookeeper-{{ cluster['name'] }} centralized coordination service"
  2. start on runlevel [2345]
  3. stop on runlevel [!2345]
  4. respawn
  5. limit nofile 8192 8192
  6. pre-start script
  7. [ -r "/etc/zookeeper-{{ cluster['name'] }}/conf/environment" ] || exit 0
  8. . /etc/zookeeper-{{ cluster['name'] }}/conf/environment
  9. [ -d $ZOO_LOG_DIR ] || mkdir -p $ZOO_LOG_DIR
  10. chown $USER:$GROUP $ZOO_LOG_DIR
  11. end script
  12. script
  13. . /etc/zookeeper-{{ cluster['name'] }}/conf/environment
  14. [ -r /etc/default/zookeeper ] && . /etc/default/zookeeper
  15. if [ -z "$JMXDISABLE" ]; then
  16. JAVA_OPTS="$JAVA_OPTS -Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.local.only=$JMXLOCALONLY"
  17. fi
  18. exec start-stop-daemon --start -c $USER --exec $JAVA --name zookeeper-{{ cluster['name'] }} \
  19. -- -cp $CLASSPATH $JAVA_OPTS -Dzookeeper.log.dir=${ZOO_LOG_DIR} \
  20. -Dzookeeper.root.logger=${ZOO_LOG4J_PROP} $ZOOMAIN $ZOOCFG
  21. end script