2017-01-31 19:53:05 +00:00
#!/bin/sh
2008-12-08 06:42:56 +00:00
### BEGIN INIT INFO
2017-01-31 19:53:05 +00:00
# Provides: clickhouse-server
2016-05-31 02:40:03 +00:00
# Default-Start: 2 3 4 5
# Default-Stop: 0 1 6
2016-05-27 20:29:45 +00:00
# Required-Start:
# Required-Stop:
2017-01-31 19:53:05 +00:00
# Short-Description: Yandex clickhouse-server daemon
2008-12-08 06:42:56 +00:00
### END INIT INFO
2017-01-31 19:53:05 +00:00
CLICKHOUSE_USER=clickhouse
CLICKHOUSE_GROUP=${CLICKHOUSE_USER}
2014-12-11 16:55:25 +00:00
SHELL=/bin/bash
2017-01-31 19:53:05 +00:00
PROGRAM=clickhouse-server
2017-03-26 03:29:34 +00:00
GENERIC_PROGRAM=clickhouse
2018-03-01 20:16:03 +00:00
EXTRACT_FROM_CONFIG=${GENERIC_PROGRAM}-extract-from-config
2008-12-08 06:42:56 +00:00
SYSCONFDIR=/etc/$PROGRAM
2017-02-03 13:52:59 +00:00
CLICKHOUSE_LOGDIR=/var/log/clickhouse-server
2017-01-31 19:53:05 +00:00
CLICKHOUSE_LOGDIR_USER=root
CLICKHOUSE_DATADIR_OLD=/opt/clickhouse
2008-12-08 06:42:56 +00:00
LOCALSTATEDIR=/var/lock
BINDIR=/usr/bin
2017-01-31 19:53:05 +00:00
CLICKHOUSE_CRONFILE=/etc/cron.d/clickhouse-server
CLICKHOUSE_CONFIG=$SYSCONFDIR/config.xml
2008-12-08 06:42:56 +00:00
LOCKFILE=$LOCALSTATEDIR/$PROGRAM
RETVAL=0
2017-01-31 19:53:05 +00:00
CLICKHOUSE_PIDDIR=/var/run/$PROGRAM
CLICKHOUSE_PIDFILE="$CLICKHOUSE_PIDDIR/$PROGRAM.pid"
2014-12-09 12:44:50 +00:00
2018-05-16 00:34:56 +00:00
# Some systems lack "flock"
command -v flock >/dev/null && FLOCK=flock
2017-02-26 18:15:48 +00:00
2017-01-31 19:53:05 +00:00
# Override defaults from optional config file
test -f /etc/default/clickhouse && . /etc/default/clickhouse
2014-12-09 12:44:50 +00:00
2016-06-10 13:07:37 +00:00
# On x86_64, check for required instruction set.
2016-06-10 19:36:35 +00:00
if uname -mpi | grep -q 'x86_64'; then
2017-07-08 16:09:32 +00:00
if ! grep -q 'sse4_2' /proc/cpuinfo; then
# On KVM, cpuinfo could falsely not report SSE 4.2 support, so skip the check.
if ! grep -q 'Common KVM processor' /proc/cpuinfo; then
# Some other VMs also report wrong flags in cpuinfo.
# Tricky way to test for instruction set:
# create temporary binary and run it;
# if it get caught illegal instruction signal,
# then required instruction set is not supported really.
#
# Generated this way:
# gcc -xc -Os -static -nostdlib - <<< 'void _start() { __asm__("pcmpgtq %%xmm0, %%xmm1; mov $0x3c, %%rax; xor %%rdi, %%rdi; syscall":::"memory"); }' && strip -R .note.gnu.build-id -R .comment -R .eh_frame -s ./a.out && gzip -c -9 ./a.out | base64 -w0; echo
if ! (echo -n 'H4sICAwAW1cCA2Eub3V0AKt39XFjYmRkgAEmBjsGEI+H0QHMd4CKGyCUAMUsGJiBJDNQNUiYlQEZOKDQclB9cnD9CmCSBYqJBRxQOvBpSQobGfqIAWn8FuYnPI4fsAGyPQz/87MeZtArziguKSpJTGLQK0mtKGGgGHADMSgoYH6AhTMPNHyE0NQzYuEzYzEXFr6CBPQDANAsXKTwAQAA' | base64 -d | gzip -d > /tmp/clickhouse_test_sse42 && chmod a+x /tmp/clickhouse_test_sse42 && /tmp/clickhouse_test_sse42); then
echo 'Warning! SSE 4.2 instruction set is not supported'
#exit 3
fi
fi
fi
2016-06-10 13:07:37 +00:00
fi
2017-02-26 18:15:48 +00:00
2017-01-31 19:53:05 +00:00
SUPPORTED_COMMANDS="{start|stop|status|restart|forcestop|forcerestart|reload|condstart|condstop|condrestart|condreload|initdb}"
2015-03-25 11:36:46 +00:00
is_supported_command()
{
2017-07-08 16:09:32 +00:00
echo "$SUPPORTED_COMMANDS" | grep -E "(\{|\|)$1(\||})" &> /dev/null
2015-03-25 11:36:46 +00:00
}
2014-11-28 16:47:02 +00:00
is_running()
{
2017-07-08 16:09:32 +00:00
[ -r "$CLICKHOUSE_PIDFILE" ] && pgrep -s $(cat "$CLICKHOUSE_PIDFILE") 1> /dev/null 2> /dev/null
2014-11-28 16:47:02 +00:00
}
2008-12-08 06:42:56 +00:00
2017-02-26 18:15:48 +00:00
wait_for_done()
2015-02-10 11:35:14 +00:00
{
2017-07-08 16:09:32 +00:00
while is_running; do
sleep 1
done
2008-12-08 06:42:56 +00:00
}
2017-01-31 19:53:05 +00:00
2017-03-26 03:29:34 +00:00
die()
{
2017-07-08 16:09:32 +00:00
echo $1 >&2
exit 1
2017-03-26 03:29:34 +00:00
}
# Check that configuration file is Ok.
check_config()
{
2018-03-01 20:16:03 +00:00
if [ -x "$BINDIR/$EXTRACT_FROM_CONFIG" ]; then
su -s $SHELL ${CLICKHOUSE_USER} -c "$BINDIR/$EXTRACT_FROM_CONFIG --config-file=\"$CLICKHOUSE_CONFIG\" --key=path" >/dev/null || die "Configuration file ${CLICKHOUSE_CONFIG} doesn't parse successfully. Won't restart server. You may use forcerestart if you are sure.";
2017-07-08 16:09:32 +00:00
fi
2017-03-26 03:29:34 +00:00
}
2017-01-31 19:53:05 +00:00
initdb()
{
2017-10-11 19:11:51 +00:00
if [ -d ${SYSCONFDIR} ]; then
su -s /bin/sh ${CLICKHOUSE_USER} -c "test -w ${SYSCONFDIR}" || chown ${CLICKHOUSE_USER}:${CLICKHOUSE_GROUP} ${SYSCONFDIR}
fi
2018-03-01 20:16:03 +00:00
if [ -x "$BINDIR/$EXTRACT_FROM_CONFIG" ]; then
CLICKHOUSE_DATADIR_FROM_CONFIG=$(su -s $SHELL ${CLICKHOUSE_USER} -c "$BINDIR/$EXTRACT_FROM_CONFIG --config-file=\"$CLICKHOUSE_CONFIG\" --key=path")
2017-07-08 16:09:32 +00:00
if [ "(" "$?" -ne "0" ")" -o "(" -z "${CLICKHOUSE_DATADIR_FROM_CONFIG}" ")" ]; then
die "Cannot obtain value of path from config file: ${CLICKHOUSE_CONFIG}";
fi
echo "Path to data directory in ${CLICKHOUSE_CONFIG}: ${CLICKHOUSE_DATADIR_FROM_CONFIG}"
else
CLICKHOUSE_DATADIR_FROM_CONFIG="/var/lib/clickhouse"
fi
if ! getent group ${CLICKHOUSE_USER} >/dev/null; then
echo "Can't chown to non-existing user ${CLICKHOUSE_USER}"
return
fi
if ! getent passwd ${CLICKHOUSE_GROUP} >/dev/null; then
echo "Can't chown to non-existing group ${CLICKHOUSE_GROUP}"
return
fi
if ! $(su -s $SHELL ${CLICKHOUSE_USER} -c "test -r ${CLICKHOUSE_CONFIG}"); then
echo "Warning! clickhouse config [${CLICKHOUSE_CONFIG}] not readable by user [${CLICKHOUSE_USER}]"
fi
if ! $(su -s $SHELL ${CLICKHOUSE_USER} -c "test -O \"${CLICKHOUSE_DATADIR_FROM_CONFIG}\" && test -G \"${CLICKHOUSE_DATADIR_FROM_CONFIG}\""); then
if [ $(dirname "${CLICKHOUSE_DATADIR_FROM_CONFIG}") == "/" ]; then
echo "Directory ${CLICKHOUSE_DATADIR_FROM_CONFIG} seems too dangerous to chown."
else
if [ ! -e "${CLICKHOUSE_DATADIR_FROM_CONFIG}" ]; then
echo "Creating directory ${CLICKHOUSE_DATADIR_FROM_CONFIG}"
mkdir -p "${CLICKHOUSE_DATADIR_FROM_CONFIG}"
fi
echo "Changing owner of [${CLICKHOUSE_DATADIR_FROM_CONFIG}] to [${CLICKHOUSE_USER}:${CLICKHOUSE_GROUP}]"
chown -R ${CLICKHOUSE_USER}:${CLICKHOUSE_GROUP} "${CLICKHOUSE_DATADIR_FROM_CONFIG}"
fi
fi
if ! $(su -s $SHELL ${CLICKHOUSE_USER} -c "test -w ${CLICKHOUSE_LOGDIR}"); then
echo "Changing owner of [${CLICKHOUSE_LOGDIR}/*] to [${CLICKHOUSE_USER}:${CLICKHOUSE_GROUP}]"
chown -R ${CLICKHOUSE_USER}:${CLICKHOUSE_GROUP} ${CLICKHOUSE_LOGDIR}/*
echo "Changing owner of [${CLICKHOUSE_LOGDIR}] to [${CLICKHOUSE_LOGDIR_USER}:${CLICKHOUSE_GROUP}]"
chown ${CLICKHOUSE_LOGDIR_USER}:${CLICKHOUSE_GROUP} ${CLICKHOUSE_LOGDIR}
fi
2017-01-31 19:53:05 +00:00
}
2017-02-26 18:15:48 +00:00
2008-12-08 06:42:56 +00:00
start()
{
2017-07-08 16:09:32 +00:00
[ -x $BINDIR/$PROGRAM ] || exit 0
local EXIT_STATUS
EXIT_STATUS=0
echo -n "Start $PROGRAM service: "
ulimit -n 262144
if is_running; then
echo -n "already running "
EXIT_STATUS=1
else
mkdir -p $CLICKHOUSE_PIDDIR
chown -R $CLICKHOUSE_USER:$CLICKHOUSE_GROUP $CLICKHOUSE_PIDDIR
2018-05-17 16:01:41 +00:00
# Set net_admin capabilities to support ClickHouse better introspection
2018-08-22 21:00:52 +00:00
setcap cap_net_admin=+ep "$BINDIR/$GENERIC_PROGRAM"
2017-07-08 16:09:32 +00:00
initdb
if ! is_running; then
# Lock should not be held while running child process, so we release the lock. Note: obviously, there is race condition.
# But clickhouse-server has protection from simultaneous runs with same data directory.
2018-05-16 00:34:56 +00:00
su -s $SHELL ${CLICKHOUSE_USER} -c "$FLOCK -u 9; exec -a \"$PROGRAM\" \"$BINDIR/$PROGRAM\" --daemon --pid-file=\"$CLICKHOUSE_PIDFILE\" --config-file=\"$CLICKHOUSE_CONFIG\""
2017-07-08 16:09:32 +00:00
EXIT_STATUS=$?
if [ $EXIT_STATUS -ne 0 ]; then
break
fi
fi
fi
if [ $EXIT_STATUS -eq 0 ]; then
echo "DONE"
else
echo "FAILED"
fi
return $EXIT_STATUS
2008-12-08 06:42:56 +00:00
}
2017-02-26 18:15:48 +00:00
2008-12-08 06:42:56 +00:00
stop()
{
2017-07-08 16:09:32 +00:00
local EXIT_STATUS
EXIT_STATUS=0
2008-12-08 06:42:56 +00:00
2017-07-08 16:09:32 +00:00
if [ -f $CLICKHOUSE_PIDFILE ]; then
2008-12-08 06:42:56 +00:00
2017-07-08 16:09:32 +00:00
echo -n "Stop $PROGRAM service: "
2014-12-29 17:31:30 +00:00
2017-07-08 16:09:32 +00:00
kill -TERM $(cat "$CLICKHOUSE_PIDFILE")
2008-12-08 06:42:56 +00:00
2017-07-08 16:09:32 +00:00
wait_for_done
2017-02-03 13:52:59 +00:00
2017-07-08 16:09:32 +00:00
echo "DONE"
fi
return $EXIT_STATUS
2008-12-08 06:42:56 +00:00
}
2017-02-26 18:15:48 +00:00
2008-12-08 06:42:56 +00:00
restart()
{
2017-07-08 16:09:32 +00:00
check_config
stop
start
2008-12-08 06:42:56 +00:00
}
2017-02-26 18:15:48 +00:00
2010-03-09 19:01:07 +00:00
forcestop()
{
2017-07-08 16:09:32 +00:00
local EXIT_STATUS
EXIT_STATUS=0
2010-03-09 19:01:07 +00:00
2017-07-08 16:09:32 +00:00
echo -n "Stop forcefully $PROGRAM service: "
2010-03-09 19:01:07 +00:00
2017-07-08 16:09:32 +00:00
kill -KILL $(cat "$CLICKHOUSE_PIDFILE")
2014-12-29 17:31:30 +00:00
2017-07-08 16:09:32 +00:00
wait_for_done
2010-03-09 19:01:07 +00:00
2017-07-08 16:09:32 +00:00
echo "DONE"
return $EXIT_STATUS
2010-03-09 19:01:07 +00:00
}
2017-02-26 18:15:48 +00:00
2010-03-09 19:01:07 +00:00
forcerestart()
{
2017-07-08 16:09:32 +00:00
forcestop
start
2010-03-09 19:01:07 +00:00
}
2018-03-01 20:16:03 +00:00
use_cron()
{
# 1. running systemd
if [ -x "/bin/systemctl" ] && [ -f /etc/systemd/system/clickhouse-server.service ] && [ -d /run/systemd/system ]; then
return 1
fi
# 2. disabled by config
if [ -z "$CLICKHOUSE_CRONFILE" ]; then
return 2
fi
return 0
}
2017-02-26 18:15:48 +00:00
2009-04-03 14:48:11 +00:00
enable_cron()
{
2018-03-01 20:16:03 +00:00
use_cron && sed -i 's/^#*//' "$CLICKHOUSE_CRONFILE"
2009-04-03 14:48:11 +00:00
}
2017-02-26 18:15:48 +00:00
2009-04-03 14:48:11 +00:00
disable_cron()
{
2018-03-01 20:16:03 +00:00
use_cron && sed -i 's/^#*/#/' "$CLICKHOUSE_CRONFILE"
2009-04-03 14:48:11 +00:00
}
2017-02-26 18:15:48 +00:00
2014-12-29 17:31:30 +00:00
is_cron_disabled()
{
2018-03-01 20:16:03 +00:00
use_cron || return 0
2017-02-26 17:59:42 +00:00
2017-07-08 16:09:32 +00:00
# Assumes that either no lines are commented or all lines are commented.
# Also please note, that currently cron file for ClickHouse has only one line (but some time ago there was more).
grep -q -E '^#' "$CLICKHOUSE_CRONFILE";
2014-12-29 17:31:30 +00:00
}
2017-02-26 18:15:48 +00:00
2014-12-29 17:31:30 +00:00
main()
{
2017-07-08 16:09:32 +00:00
# See how we were called.
EXIT_STATUS=0
case "$1" in
start)
start && enable_cron
;;
stop)
disable_cron && stop
;;
restart)
restart && enable_cron
;;
forcestop)
disable_cron && forcestop
;;
forcerestart)
forcerestart && enable_cron
;;
reload)
restart
;;
condstart)
is_running || start
;;
condstop)
is_running && stop
;;
condrestart)
is_running && restart
;;
condreload)
is_running && restart
;;
initdb)
initdb
;;
2018-02-01 17:55:08 +00:00
enable_cron)
enable_cron
;;
disable_cron)
disable_cron
;;
2017-07-08 16:09:32 +00:00
*)
echo "Usage: $0 $SUPPORTED_COMMANDS"
exit 2
;;
esac
exit $EXIT_STATUS
2014-12-29 17:31:30 +00:00
}
2008-12-08 06:42:56 +00:00
2017-02-26 18:15:48 +00:00
2015-03-24 15:28:02 +00:00
status()
{
2017-07-08 16:09:32 +00:00
if is_running; then
echo "$PROGRAM service is running"
else
if is_cron_disabled; then
echo "$PROGRAM service is stopped";
else
echo "$PROGRAM: process unexpectedly terminated"
fi
fi
2015-03-24 15:28:02 +00:00
}
2017-02-26 18:15:48 +00:00
2017-02-15 23:43:52 +00:00
# Running commands without need of locking
case "$1" in
status)
2017-07-08 16:09:32 +00:00
status
exit 0
;;
2017-02-15 23:43:52 +00:00
esac
2015-03-25 11:36:46 +00:00
2017-02-26 18:15:48 +00:00
2014-12-29 17:31:30 +00:00
(
2018-05-16 00:34:56 +00:00
if $FLOCK -n 9; then
2017-07-08 16:09:32 +00:00
main "$@"
else
echo "Init script is already running" && exit 1
fi
2017-02-26 18:53:24 +00:00
) 9> $LOCKFILE