ceph: add mgr-restful-plugin service
ceph-rest-api was removed in Ceph Mimic. Similar functionality is provided by ceph-mgr restful plugin. This new service needs to be configured and started instead of ceph-rest-api: * enable mgr-restful-plugin in ceph.conf * add mgr-restrul-plugin service file * add mgr-restful-plugin monitoring daemon waiting for ceph to become available before it configures and starts ceph-mgr daemon providing RESTful API endpoints Story: 2003605 Task: 28860 Depends-On: Iaa3319a7647e5622037d12c53673da0e4199ceb4 Change-Id: I31fb9aac89c44bbce24939197446caa987d395cb Co-Authored-By: Yong Hu <yong.hu@intel.com> Co-Authored-By: Changcheng Liu <changcheng.liu@intel.com> Signed-off-by: Daniel Badea <daniel.badea@windriver.com>
This commit is contained in:
parent
e6ddff6eb0
commit
5fe6c52773
@ -290,7 +290,7 @@ Source17: src/spdk/dpdk/dpdk-6ece49ad5a26f5e2f5c4af6c06c30376c0ddc387.tar.gz
|
|||||||
Source18: src/rapidjson/thirdparty/gtest/googletest-0a439623f75c029912728d80cb7f1b8b48739ca4.tar.gz
|
Source18: src/rapidjson/thirdparty/gtest/googletest-0a439623f75c029912728d80cb7f1b8b48739ca4.tar.gz
|
||||||
|
|
||||||
Source19: ceph.sh
|
Source19: ceph.sh
|
||||||
Source20: mgr-restful-plugin
|
Source20: mgr-restful-plugin.py
|
||||||
Source21: ceph.conf.pmon
|
Source21: ceph.conf.pmon
|
||||||
Source22: ceph-init-wrapper.sh
|
Source22: ceph-init-wrapper.sh
|
||||||
Source23: ceph.conf
|
Source23: ceph.conf
|
||||||
@ -1292,7 +1292,7 @@ mkdir -p %{buildroot}%{_unitdir}
|
|||||||
install -m 750 %{SOURCE19} %{buildroot}%{_sysconfdir}/services.d/controller/
|
install -m 750 %{SOURCE19} %{buildroot}%{_sysconfdir}/services.d/controller/
|
||||||
install -m 750 %{SOURCE19} %{buildroot}%{_sysconfdir}/services.d/storage/
|
install -m 750 %{SOURCE19} %{buildroot}%{_sysconfdir}/services.d/storage/
|
||||||
install -m 750 %{SOURCE19} %{buildroot}%{_sysconfdir}/services.d/worker/
|
install -m 750 %{SOURCE19} %{buildroot}%{_sysconfdir}/services.d/worker/
|
||||||
install -m 750 %{SOURCE20} %{buildroot}%{_initrddir}/
|
install -m 750 %{SOURCE20} %{buildroot}%{_initrddir}/mgr-restful-plugin
|
||||||
install -m 750 %{SOURCE21} %{buildroot}%{_sysconfdir}/ceph/
|
install -m 750 %{SOURCE21} %{buildroot}%{_sysconfdir}/ceph/
|
||||||
install -m 750 %{SOURCE22} %{buildroot}%{_initrddir}/ceph-init-wrapper
|
install -m 750 %{SOURCE22} %{buildroot}%{_initrddir}/ceph-init-wrapper
|
||||||
install -m 640 %{SOURCE23} %{buildroot}%{_sysconfdir}/ceph/
|
install -m 640 %{SOURCE23} %{buildroot}%{_sysconfdir}/ceph/
|
||||||
|
@ -48,3 +48,4 @@
|
|||||||
# Quiet new warnings on move to Hammer
|
# Quiet new warnings on move to Hammer
|
||||||
mon pg warn max per osd = 2048
|
mon pg warn max per osd = 2048
|
||||||
mon pg warn max object skew = 0
|
mon pg warn max object skew = 0
|
||||||
|
mgr initial modules = restful
|
||||||
|
753
ceph/ceph/files/mgr-restful-plugin.py
Normal file
753
ceph/ceph/files/mgr-restful-plugin.py
Normal file
@ -0,0 +1,753 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
#
|
||||||
|
# Copyright (c) 2019 Wind River Systems, Inc.
|
||||||
|
#
|
||||||
|
# SPDX-License-Identifier: Apache-2.0
|
||||||
|
#
|
||||||
|
|
||||||
|
|
||||||
|
### BEGIN INIT INFO
|
||||||
|
# Provides: ceph/mgr RESTful API plugin
|
||||||
|
# Required-Start: $ceph
|
||||||
|
# Required-Stop: $ceph
|
||||||
|
# Default-Start: 2 3 4 5
|
||||||
|
# Default-Stop: 0 1 6
|
||||||
|
# Short-Description: Ceph MGR RESTful API plugin
|
||||||
|
# Description: Ceph MGR RESTful API plugin
|
||||||
|
### END INIT INFO
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import errno
|
||||||
|
import fcntl
|
||||||
|
import inspect
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import signal
|
||||||
|
import socket
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import daemon
|
||||||
|
import psutil
|
||||||
|
import requests
|
||||||
|
|
||||||
|
|
||||||
|
class Config(object):
|
||||||
|
|
||||||
|
"""Configuration options controlling ceph-mgr restful plugin service wrapper. In the future we may want to load them from a configuration file (like /etc/ceph/mgr-restful-plugin.conf ) """
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.service_name = 'mgr-restful-plugin'
|
||||||
|
self.pgrep_pattern = os.path.join(
|
||||||
|
'/etc/init.d/', self.service_name)
|
||||||
|
self.log_level = logging.INFO
|
||||||
|
self.log_dir = '/var/log'
|
||||||
|
self.ceph_mgr_service = '/usr/bin/ceph-mgr'
|
||||||
|
self.ceph_mgr_cluster = 'ceph'
|
||||||
|
self.ceph_mgr_rundir = '/var/run/ceph/mgr'
|
||||||
|
self.service_socket = os.path.join(
|
||||||
|
self.ceph_mgr_rundir,
|
||||||
|
'{}.socket'.format(self.service_name))
|
||||||
|
self.service_lock = os.path.join(
|
||||||
|
self.ceph_mgr_rundir,
|
||||||
|
'{}.lock'.format(self.service_name))
|
||||||
|
self.service_pidfile = os.path.join(
|
||||||
|
'/var/run/ceph',
|
||||||
|
'{}.pid'.format(self.service_name))
|
||||||
|
self.ceph_name = socket.gethostname()
|
||||||
|
self.service_port = 5001
|
||||||
|
self.timeout_seconds = 15
|
||||||
|
self.cluster_retry_sec = 15
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def load():
|
||||||
|
return Config()
|
||||||
|
|
||||||
|
|
||||||
|
CONFIG = Config.load()
|
||||||
|
LOG = None
|
||||||
|
|
||||||
|
|
||||||
|
def setup_logging(name=None, cleanup_handlers=False):
|
||||||
|
global LOG
|
||||||
|
if not name:
|
||||||
|
name = CONFIG.service_name
|
||||||
|
LOG = logging.getLogger(name)
|
||||||
|
LOG.setLevel(CONFIG.log_level)
|
||||||
|
if cleanup_handlers:
|
||||||
|
try:
|
||||||
|
for handler in LOG.handlers:
|
||||||
|
if isinstance(handler, logging.StreamHandler):
|
||||||
|
handler.flush()
|
||||||
|
if isinstance(handler, logging.FileHandler):
|
||||||
|
handler.close()
|
||||||
|
LOG.handlers = []
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
elif LOG.handlers:
|
||||||
|
return
|
||||||
|
handler = logging.FileHandler(
|
||||||
|
os.path.join(CONFIG.log_dir,
|
||||||
|
'{}.log'.format(CONFIG.service_name)))
|
||||||
|
handler.setFormatter(
|
||||||
|
logging.Formatter('%(asctime)s %(process)s %(levelname)s %(name)s %(message)s'))
|
||||||
|
LOG.addHandler(handler)
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceException(Exception):
|
||||||
|
message = "generic ceph mgr service exception"
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
if "message" not in kwargs:
|
||||||
|
try:
|
||||||
|
message = self.message.format(*args, **kwargs)
|
||||||
|
except Exception: # noqa
|
||||||
|
message = '{}, args:{}, kwargs: {}'.format(
|
||||||
|
self.message, args, kwargs)
|
||||||
|
else:
|
||||||
|
message = kwargs["message"]
|
||||||
|
super(ServiceException, self).__init__(message)
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMonitorAlreadyStarted(ServiceException):
|
||||||
|
message = ('Service monitor already started')
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMonitorLockFailed(ServiceException):
|
||||||
|
message = ('Unable to lock service monitor: '
|
||||||
|
'reason={reason}')
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMonitorNoSocket(ServiceException):
|
||||||
|
message = ('Unable to create monitor socket: '
|
||||||
|
'reason={reason}')
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMonitorBindFailed(ServiceException):
|
||||||
|
message = ('Failed to bind monitor socket: '
|
||||||
|
'path={path}, reason={reason}')
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMonitor(object):
|
||||||
|
def __init__(self):
|
||||||
|
LOG.info('setup service monitor')
|
||||||
|
self.plugin_started = False
|
||||||
|
self.service_url = None
|
||||||
|
self.child = None
|
||||||
|
self.consecutive_ceph_mgr_failures = 0
|
||||||
|
self.consecutive_restful_plugin_restarts = 0
|
||||||
|
|
||||||
|
LOG.info('setup signal handler')
|
||||||
|
signal.signal(signal.SIGCHLD, self.handle_child_signal)
|
||||||
|
|
||||||
|
LOG.info('take service lock')
|
||||||
|
try:
|
||||||
|
os.makedirs(os.path.dirname(CONFIG.service_lock))
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
self.lock_file = open(CONFIG.service_lock, 'w')
|
||||||
|
try:
|
||||||
|
fcntl.flock(self.lock_file.fileno(),
|
||||||
|
fcntl.LOCK_EX | fcntl.LOCK_NB)
|
||||||
|
except (IOError, OSError) as err:
|
||||||
|
if err.errno == errno.EAGAIN:
|
||||||
|
raise ServiceMonitorAlreadyStarted()
|
||||||
|
else:
|
||||||
|
raise ServiceMonitorLockFailed(reason=str(err))
|
||||||
|
|
||||||
|
LOG.info('create service socket')
|
||||||
|
try:
|
||||||
|
self.socket = socket.socket(
|
||||||
|
socket.AF_UNIX, socket.SOCK_STREAM)
|
||||||
|
except socket.error as err:
|
||||||
|
raise ServiceMonitorNoSocket(reason=err)
|
||||||
|
|
||||||
|
LOG.info('remove existing service socket files (if any)')
|
||||||
|
try:
|
||||||
|
os.unlink(CONFIG.service_socket)
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
LOG.info('bind service socket: path=%s',
|
||||||
|
CONFIG.service_socket)
|
||||||
|
try:
|
||||||
|
self.socket.bind(CONFIG.service_socket)
|
||||||
|
except socket.error as err:
|
||||||
|
raise ServiceMonitorBindFailed(
|
||||||
|
path=CONFIG.service_socket, reason=err)
|
||||||
|
|
||||||
|
LOG.info('set service socket timeout')
|
||||||
|
self.socket.settimeout(1)
|
||||||
|
|
||||||
|
LOG.info('clear service state machine')
|
||||||
|
self.state = None
|
||||||
|
self.skip_ticks = None
|
||||||
|
|
||||||
|
def handle_child_signal(self, signum, frame):
|
||||||
|
if signum != signal.SIGCHLD:
|
||||||
|
LOG.debug('discarding unexpected signal %d', signum)
|
||||||
|
return
|
||||||
|
if not self.child:
|
||||||
|
LOG.debug('child service not started; discard signal', signum)
|
||||||
|
return
|
||||||
|
pid, status = os.waitpid(-1, os.WNOHANG | os.WUNTRACED | os.WCONTINUED)
|
||||||
|
if pid == 0 and status == 0:
|
||||||
|
LOG.debug('child status unavailable')
|
||||||
|
return
|
||||||
|
if pid != self.child.pid:
|
||||||
|
LOG.debug('signal pid %d does not match expected pid %d',
|
||||||
|
pid, self.child.pid)
|
||||||
|
return
|
||||||
|
if os.WIFEXITED(status) or os.WIFSIGNALED(status):
|
||||||
|
LOG.info('child has exited')
|
||||||
|
self.child = None
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
LOG.info('create service pid file')
|
||||||
|
with open(CONFIG.service_pidfile, 'w') as pid_file:
|
||||||
|
pid_file.write(str(os.getpid()))
|
||||||
|
LOG.info('start service monitor loop')
|
||||||
|
try:
|
||||||
|
self.socket.listen(1)
|
||||||
|
self.reset_monitor()
|
||||||
|
while True:
|
||||||
|
self.step_monitor()
|
||||||
|
try:
|
||||||
|
client, _ = self.socket.accept()
|
||||||
|
try:
|
||||||
|
request = client.recv(1024)
|
||||||
|
LOG.debug('received client request=%s', request)
|
||||||
|
except socket.timeout:
|
||||||
|
LOG.warning('timeout waiting for client request')
|
||||||
|
client.close()
|
||||||
|
continue
|
||||||
|
if request == 'status':
|
||||||
|
status = self.status()
|
||||||
|
if (not status.startswith('OK')
|
||||||
|
and not self.state):
|
||||||
|
LOG.info('starting automatic recovery. '
|
||||||
|
'status=%s', status)
|
||||||
|
self.reset_monitor()
|
||||||
|
try:
|
||||||
|
client.sendall(status)
|
||||||
|
except socket.error as err:
|
||||||
|
LOG.warning('failed to report status. '
|
||||||
|
'status=%s, reason=%s', status, err)
|
||||||
|
self.reset_monitor()
|
||||||
|
if request == 'stop':
|
||||||
|
self.stop()
|
||||||
|
try:
|
||||||
|
client.sendall('OK')
|
||||||
|
except socket.error as err:
|
||||||
|
LOG.warning('failed to report status. '
|
||||||
|
'status=%s, reason=%s', status, err)
|
||||||
|
LOG.info('exit service monitor loop')
|
||||||
|
return
|
||||||
|
except socket.timeout:
|
||||||
|
pass
|
||||||
|
finally:
|
||||||
|
self.socket.close()
|
||||||
|
self.socket = None
|
||||||
|
|
||||||
|
def reset_monitor(self):
|
||||||
|
self.stop_ceph_mgr_service()
|
||||||
|
self.state = self.state_wait_cluster_available
|
||||||
|
self.skip_ticks = 0
|
||||||
|
|
||||||
|
def step_monitor(self):
|
||||||
|
state = self.state.__name__ if self.state else 'none'
|
||||||
|
LOG.debug('monitor service tick. state=%s, skip_ticks=%d'
|
||||||
|
'consecutive_ceph_mgr_failures=%d, '
|
||||||
|
'consecutive_restful_plugin_restarts=%d, ',
|
||||||
|
state, self.skip_ticks,
|
||||||
|
self.consecutive_ceph_mgr_failures,
|
||||||
|
self.consecutive_restful_plugin_restarts)
|
||||||
|
if not self.state:
|
||||||
|
return
|
||||||
|
if self.skip_ticks:
|
||||||
|
self.skip_ticks -= 1
|
||||||
|
return
|
||||||
|
self.state, self.skip_ticks = self.state()
|
||||||
|
|
||||||
|
def state_wait_cluster_available(self):
|
||||||
|
LOG.info('wait until cluster is available')
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'fsid'],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.info('monitor: ceph fsid failed (cluster unavailable). '
|
||||||
|
'reason=%s, out=%s', err, err.output)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
return (self.state_stop_unmanaged_ceph_mgr, 0)
|
||||||
|
|
||||||
|
def state_stop_unmanaged_ceph_mgr(self):
|
||||||
|
self.stop_unmanaged_ceph_mgr()
|
||||||
|
return (self.state_get_ceph_mgr_auth, 0)
|
||||||
|
|
||||||
|
def stop_unmanaged_ceph_mgr(self):
|
||||||
|
LOG.info('stop any unmanaged ceph-mgr processes')
|
||||||
|
service_name = os.path.basename(CONFIG.ceph_mgr_service)
|
||||||
|
self.child = None
|
||||||
|
procs = [proc for proc in psutil.process_iter()
|
||||||
|
if proc.name() == service_name]
|
||||||
|
for proc in procs:
|
||||||
|
proc.terminate()
|
||||||
|
_, procs = psutil.wait_procs(procs, timeout=5)
|
||||||
|
for proc in procs:
|
||||||
|
proc.kill()
|
||||||
|
|
||||||
|
def state_cleanup_ceph_mgr_auth(self):
|
||||||
|
LOG.info('cleanup ceph auth for ceph-mgr')
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'auth', 'del',
|
||||||
|
'mgr.{}'.format(CONFIG.ceph_name)],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.info('monitor: ceph auth del failed. '
|
||||||
|
'reason=%s, out=%s', err, err.output)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
shutil.rmtree(CONFIG.ceph_mgr_rundir, ignore_errors=True)
|
||||||
|
return (self.state_create_ceph_mgr_auth, 0)
|
||||||
|
|
||||||
|
def state_get_ceph_mgr_auth(self):
|
||||||
|
path = '{}/ceph-{}'.format(CONFIG.ceph_mgr_rundir,
|
||||||
|
CONFIG.ceph_name)
|
||||||
|
LOG.info('create directory structure: path=%s', path)
|
||||||
|
try:
|
||||||
|
os.makedirs(path)
|
||||||
|
except OSError as err:
|
||||||
|
LOG.info('monitor: failed to create directory structure. '
|
||||||
|
'path=%s, reason=%s', path, err)
|
||||||
|
# continue running (path may already exist)
|
||||||
|
LOG.info('check ceph-mgr auth')
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'auth', 'get',
|
||||||
|
'mgr.{}'.format(CONFIG.ceph_name),
|
||||||
|
'-o', '{}/ceph-{}/keyring'.format(
|
||||||
|
CONFIG.ceph_mgr_rundir,
|
||||||
|
CONFIG.ceph_name)],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.info('missing ceph-mgr auth. '
|
||||||
|
'reason=%s, out=%s', err, err.output)
|
||||||
|
return (self.state_create_ceph_mgr_auth,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
return (self.state_start_ceph_mgr_service, 0)
|
||||||
|
|
||||||
|
def state_create_ceph_mgr_auth(self):
|
||||||
|
LOG.info('create ceph-mgr auth')
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'auth', 'get-or-create',
|
||||||
|
'mgr.{}'.format(CONFIG.ceph_name),
|
||||||
|
'mon', 'allow *', 'osd', 'allow *'],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.info('monitor: failed to create ceph-mgr auth. '
|
||||||
|
'reason=%s, out=%s', err, err.output)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
return (self.state_get_ceph_mgr_auth, 0)
|
||||||
|
|
||||||
|
def state_start_ceph_mgr_service(self):
|
||||||
|
LOG.info('start ceph-mgr daemon')
|
||||||
|
try:
|
||||||
|
null_ouput = open(os.devnull, 'wb')
|
||||||
|
self.child = psutil.Popen(
|
||||||
|
[CONFIG.ceph_mgr_service,
|
||||||
|
'--cluster', CONFIG.ceph_mgr_cluster,
|
||||||
|
'--id', CONFIG.ceph_name,
|
||||||
|
'-f'],
|
||||||
|
close_fds=True,
|
||||||
|
stdout=null_ouput,
|
||||||
|
stderr=null_ouput,
|
||||||
|
shell=False)
|
||||||
|
except OSError as err:
|
||||||
|
LOG.warning('monitor: failed to start ceph-mgr. reason=%s', err)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
return (self.state_check_restful_plugin_service, 10)
|
||||||
|
|
||||||
|
def state_check_restful_plugin_service(self):
|
||||||
|
LOG.info('check restful plugin')
|
||||||
|
try:
|
||||||
|
out = subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'mgr', 'services',
|
||||||
|
'--format', 'json'],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.warning('unable to get ceph-mgr services. '
|
||||||
|
'reason=%s, out=%s', err, err.output)
|
||||||
|
return (self.state_cleanup_ceph_mgr_auth,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
try:
|
||||||
|
services = json.loads(out)
|
||||||
|
except ValueError:
|
||||||
|
LOG.error('unable to parse services info. '
|
||||||
|
'data=%s', out)
|
||||||
|
return (None, 0)
|
||||||
|
try:
|
||||||
|
self.service_url = services['restful']
|
||||||
|
except (KeyError, TypeError):
|
||||||
|
LOG.info('restful plugin not found')
|
||||||
|
self.service_url = None
|
||||||
|
return (self.state_enable_restful_plugin, 0)
|
||||||
|
self.consecutive_restful_plugin_restarts = 0
|
||||||
|
return (self.state_ping_restful_plugin, 0)
|
||||||
|
|
||||||
|
def state_enable_restful_plugin(self):
|
||||||
|
LOG.info('enable restful plugin')
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'mgr',
|
||||||
|
'module', 'enable', 'restful'],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.warning('failed to enable restful plugin. '
|
||||||
|
'reason=%s, out=%s', err, err.output)
|
||||||
|
return (self.state_cleanup_ceph_mgr_auth,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'config', 'set', 'mgr',
|
||||||
|
'mgr/restful/server_port', str(CONFIG.service_port)],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.error('failed to configure restful plugin '
|
||||||
|
'port. reason=%s, output=%s', err, err.output)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'restful',
|
||||||
|
'create-key', 'admin'],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.error('failed to create restful admin key. '
|
||||||
|
'reason=%s, output=%s', err, err.output)
|
||||||
|
subprocess.call(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'mgr',
|
||||||
|
'module', 'disable', 'restful'],
|
||||||
|
shell=False)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'restful',
|
||||||
|
'create-self-signed-cert'],
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.error('failed to create restful self-signed-cert. '
|
||||||
|
'reason=%s, output=%s', err, err.output)
|
||||||
|
subprocess.call(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'mgr',
|
||||||
|
'module', 'disable', 'restful'],
|
||||||
|
shell=False)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
self.consecutive_restful_plugin_restarts = 0
|
||||||
|
return (self.state_ping_restful_plugin, 0)
|
||||||
|
|
||||||
|
def state_ping_restful_plugin(self):
|
||||||
|
if not self.service_url:
|
||||||
|
try:
|
||||||
|
out = subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'mgr', 'services',
|
||||||
|
'--format', 'json'],
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.warning('unable to get ceph-mgr services. '
|
||||||
|
'reason=%s', err)
|
||||||
|
return (self.state_cleanup_ceph_mgr_auth,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
try:
|
||||||
|
services = json.loads(out)
|
||||||
|
except ValueError:
|
||||||
|
LOG.error('unable to parse services info. '
|
||||||
|
'data=%s', out)
|
||||||
|
return (None, 0)
|
||||||
|
try:
|
||||||
|
self.service_url = services['restful']
|
||||||
|
except (KeyError, TypeError):
|
||||||
|
LOG.info('restful plugin not found. '
|
||||||
|
'services=%s', services)
|
||||||
|
self.service_url = None
|
||||||
|
return (self.state_enable_restful_plugin, 5)
|
||||||
|
requests.packages.urllib3.disable_warnings()
|
||||||
|
LOG.info('restful plugin url=%s', self.service_url)
|
||||||
|
try:
|
||||||
|
response = requests.request(
|
||||||
|
'GET', self.service_url,
|
||||||
|
verify=False,
|
||||||
|
timeout=CONFIG.timeout_seconds)
|
||||||
|
LOG.info('get url=%s, response=%s',
|
||||||
|
self.service_url, response)
|
||||||
|
if not response.ok:
|
||||||
|
LOG.info('restful plugin is not ok. '
|
||||||
|
'response=%s', response)
|
||||||
|
return (self.state_restart_restful_plugin, 5)
|
||||||
|
except (requests.ConnectionError,
|
||||||
|
requests.Timeout,
|
||||||
|
requests.HTTPError) as err:
|
||||||
|
LOG.error('restful request error. '
|
||||||
|
'reason=%s', err)
|
||||||
|
if not self.child:
|
||||||
|
return (self.state_start_ceph_mgr_service, 5)
|
||||||
|
return (self.state_restart_restful_plugin, 5)
|
||||||
|
self.plugin_started = True
|
||||||
|
return (self.state_check_ceph_mgr_service, 0)
|
||||||
|
|
||||||
|
def state_restart_restful_plugin(self):
|
||||||
|
try:
|
||||||
|
subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'restful', 'restart'],
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.warning('unable to get ceph-mgr services. '
|
||||||
|
'reason=%s', err)
|
||||||
|
return (self.state_wait_cluster_available,
|
||||||
|
CONFIG.cluster_retry_sec)
|
||||||
|
self.consecutive_restful_plugin_restarts += 1
|
||||||
|
LOG.info('consecutive restful plugin restarts %d',
|
||||||
|
self.consecutive_restful_plugin_restarts)
|
||||||
|
if self.consecutive_restful_plugin_restarts > 5:
|
||||||
|
self.stop_ceph_mgr_service()
|
||||||
|
self.consecutive_ceph_mgr_failures = 0
|
||||||
|
return (self.state_start_ceph_mgr_service, 5)
|
||||||
|
return (self.state_ping_restful_plugin, 5)
|
||||||
|
|
||||||
|
def state_check_ceph_mgr_service(self):
|
||||||
|
if not self.child:
|
||||||
|
self.consecutive_ceph_mgr_failures += 1
|
||||||
|
LOG.info('consecutive ceph mgr failures %d',
|
||||||
|
self.consecutive_ceph_mgr_failures)
|
||||||
|
if self.consecutive_ceph_mgr_failures > 3:
|
||||||
|
self.stop_ceph_mgr_service()
|
||||||
|
self.consecutive_ceph_mgr_failures = 0
|
||||||
|
return (self.state_start_ceph_mgr_service, 5)
|
||||||
|
self.consecutive_ceph_mgr_failures = 0
|
||||||
|
return (self.state_check_ceph_mgr_service, 0)
|
||||||
|
|
||||||
|
def status(self):
|
||||||
|
if not self.plugin_started:
|
||||||
|
if self.state and self.consecutive_ceph_mgr_failures < 5:
|
||||||
|
return 'OK.monitor.{}'.format(
|
||||||
|
self.state.__name__)
|
||||||
|
return 'ERR.no_monitor'
|
||||||
|
try:
|
||||||
|
out = subprocess.check_output(
|
||||||
|
['/usr/bin/timeout', str(CONFIG.timeout_seconds),
|
||||||
|
'/usr/bin/ceph', 'mgr', 'services',
|
||||||
|
'--format', 'json'],
|
||||||
|
shell=False)
|
||||||
|
except subprocess.CalledProcessError as err:
|
||||||
|
LOG.warning('unable to get ceph-mgr services. '
|
||||||
|
'reason=%s', err)
|
||||||
|
return 'ERR.service_unavailable'
|
||||||
|
try:
|
||||||
|
services = json.loads(out)
|
||||||
|
except ValueError:
|
||||||
|
LOG.error('unable to parse services info. '
|
||||||
|
'data=%s', out)
|
||||||
|
return 'ERR.format_error'
|
||||||
|
try:
|
||||||
|
self.service_url = services['restful']
|
||||||
|
except (KeyError, TypeError):
|
||||||
|
LOG.info('restful plugin not found. '
|
||||||
|
'services=%s', services)
|
||||||
|
return 'ERR.missing_plugin'
|
||||||
|
requests.packages.urllib3.disable_warnings()
|
||||||
|
try:
|
||||||
|
response = requests.request(
|
||||||
|
'GET', self.service_url,
|
||||||
|
verify=False,
|
||||||
|
timeout=CONFIG.timeout_seconds)
|
||||||
|
if not response.ok:
|
||||||
|
LOG.info('restful plugin is not ok. '
|
||||||
|
'response=%s', response)
|
||||||
|
return 'ERR.response_error'
|
||||||
|
except (requests.ConnectionError,
|
||||||
|
requests.Timeout,
|
||||||
|
requests.HTTPError) as err:
|
||||||
|
LOG.error('restful request error. '
|
||||||
|
'reason=%s', err)
|
||||||
|
return 'ERR.request_error'
|
||||||
|
return 'OK'
|
||||||
|
|
||||||
|
def stop_ceph_mgr_service(self):
|
||||||
|
LOG.info('stop running ceph-mgr (if any)')
|
||||||
|
# save child reference as it may be set to None
|
||||||
|
# by the signal handler
|
||||||
|
proc = self.child
|
||||||
|
if proc:
|
||||||
|
proc.terminate()
|
||||||
|
try:
|
||||||
|
proc.wait(timeout=5)
|
||||||
|
except psutil.TimeoutExpired:
|
||||||
|
proc.kill()
|
||||||
|
LOG.info('stop unmanaged ceph-mgr (if any)')
|
||||||
|
self.stop_unmanaged_ceph_mgr()
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
LOG.info('stop service monitor')
|
||||||
|
self.state = None
|
||||||
|
self.skip_ticks = 0
|
||||||
|
self.stop_ceph_mgr_service()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _make_client_socket():
|
||||||
|
sock = socket.socket(
|
||||||
|
socket.AF_UNIX, socket.SOCK_STREAM)
|
||||||
|
sock.settimeout(CONFIG.timeout_seconds)
|
||||||
|
sock.connect(CONFIG.service_socket)
|
||||||
|
return sock
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def request_status():
|
||||||
|
try:
|
||||||
|
sock = ServiceMonitor._make_client_socket()
|
||||||
|
sock.sendall('status')
|
||||||
|
status = sock.recv(1024)
|
||||||
|
LOG.debug('status %s', status)
|
||||||
|
return status.startswith('OK')
|
||||||
|
except socket.error as err:
|
||||||
|
LOG.error('status error: reason=%s', err)
|
||||||
|
return False
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def request_stop():
|
||||||
|
try:
|
||||||
|
sock = ServiceMonitor._make_client_socket()
|
||||||
|
sock.sendall('stop')
|
||||||
|
sock.recv(1024)
|
||||||
|
except socket.error as err:
|
||||||
|
LOG.error('stop error: reason=%s', err)
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class InitWrapper(object):
|
||||||
|
def __init__(self):
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
actions = [m[0]
|
||||||
|
for m in inspect.getmembers(self)
|
||||||
|
if (inspect.ismethod(m[1])
|
||||||
|
and m[0] not in ['__init__', 'run'])]
|
||||||
|
parser.add_argument(
|
||||||
|
'action',
|
||||||
|
choices=actions)
|
||||||
|
self.args = parser.parse_args()
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
getattr(self, self.args.action)()
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
pipe = os.pipe()
|
||||||
|
child = os.fork()
|
||||||
|
if child == 0:
|
||||||
|
os.close(pipe[0])
|
||||||
|
with daemon.DaemonContext(files_preserve=[pipe[1]]):
|
||||||
|
setup_logging(cleanup_handlers=True)
|
||||||
|
try:
|
||||||
|
monitor = ServiceMonitor()
|
||||||
|
status = 'OK'
|
||||||
|
except ServiceMonitorAlreadyStarted:
|
||||||
|
os.write(pipe[1], 'OK')
|
||||||
|
os.close(pipe[1])
|
||||||
|
return
|
||||||
|
except Exception as err:
|
||||||
|
status = str(err)
|
||||||
|
os.write(pipe[1], status)
|
||||||
|
os.close(pipe[1])
|
||||||
|
if status == 'OK':
|
||||||
|
try:
|
||||||
|
monitor.run()
|
||||||
|
except Exception as err:
|
||||||
|
LOG.exception('service monitor error: '
|
||||||
|
'reason=%s', err)
|
||||||
|
else:
|
||||||
|
os.close(pipe[1])
|
||||||
|
try:
|
||||||
|
status = os.read(pipe[0], 1024)
|
||||||
|
if status == 'OK':
|
||||||
|
sys.exit(0)
|
||||||
|
else:
|
||||||
|
LOG.warning('Service monitor failed to start: '
|
||||||
|
'status=%s', status)
|
||||||
|
except IOError as err:
|
||||||
|
LOG.warning('Failed to read monitor status: '
|
||||||
|
'reason=%s', err)
|
||||||
|
os.close(pipe[0])
|
||||||
|
os.waitpid(child, 0)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
result = ServiceMonitor.request_stop()
|
||||||
|
if not result:
|
||||||
|
ceph_mgr = os.path.basename(CONFIG.ceph_mgr_service)
|
||||||
|
procs = []
|
||||||
|
for proc in psutil.process_iter():
|
||||||
|
name = proc.name()
|
||||||
|
if name == CONFIG.service_name:
|
||||||
|
procs.append(proc)
|
||||||
|
if name == ceph_mgr:
|
||||||
|
procs.append(proc)
|
||||||
|
for proc in procs:
|
||||||
|
proc.terminate()
|
||||||
|
_, procs = psutil.wait_procs(procs, timeout=5)
|
||||||
|
for proc in procs:
|
||||||
|
proc.kill()
|
||||||
|
|
||||||
|
def restart(self):
|
||||||
|
self.stop()
|
||||||
|
self.start()
|
||||||
|
|
||||||
|
def force_reload(self):
|
||||||
|
self.stop()
|
||||||
|
self.start()
|
||||||
|
|
||||||
|
def reload(self):
|
||||||
|
self.stop()
|
||||||
|
self.start()
|
||||||
|
|
||||||
|
def status(self):
|
||||||
|
status = ServiceMonitor.request_status()
|
||||||
|
sys.exit(0 if status is True else 1)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
setup_logging(name='init-wrapper')
|
||||||
|
InitWrapper().run()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
15
ceph/ceph/files/mgr-restful-plugin.service
Normal file
15
ceph/ceph/files/mgr-restful-plugin.service
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Ceph MGR RESTful API Plugin
|
||||||
|
After=network-online.target sw-patch.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=forking
|
||||||
|
Restart=no
|
||||||
|
KillMode=process
|
||||||
|
RemainAfterExit=yes
|
||||||
|
ExecStart=/etc/rc.d/init.d/mgr-restful-plugin start
|
||||||
|
ExecStop=/etc/rc.d/init.d/mgr-restul-plugin stop
|
||||||
|
ExecReload=/etc/rc.d/init.d/mgr-restful-plugin reload
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
Loading…
Reference in New Issue
Block a user