2013-12-04 23:54:29 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
""" License
|
|
|
|
|
|
|
|
Copyright (C) 2013 YunoHost
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Affero General Public License as published
|
|
|
|
by the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
along with this program; if not, see http://www.gnu.org/licenses
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
2013-12-05 01:15:34 +01:00
|
|
|
""" yunohost_service.py
|
2013-12-04 23:54:29 +01:00
|
|
|
|
2013-12-05 01:15:34 +01:00
|
|
|
Manage services
|
2013-12-04 23:54:29 +01:00
|
|
|
"""
|
2014-03-25 18:56:51 +01:00
|
|
|
import logging
|
|
|
|
logging.warning('the module yunohost.service has not been revisited and updated yet')
|
|
|
|
|
2013-12-04 23:54:29 +01:00
|
|
|
import yaml
|
|
|
|
import glob
|
|
|
|
import subprocess
|
|
|
|
import os.path
|
2014-02-05 02:01:03 +01:00
|
|
|
|
2014-03-04 12:31:04 +01:00
|
|
|
from moulinette.helpers import YunoHostError, win_msg
|
2013-12-04 23:54:29 +01:00
|
|
|
|
|
|
|
|
|
|
|
def service_start(names):
|
|
|
|
"""
|
|
|
|
Start one or more services
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
names -- Services name to start
|
|
|
|
|
|
|
|
"""
|
2013-12-17 00:26:27 +01:00
|
|
|
if isinstance(names, str):
|
|
|
|
names = [names]
|
2013-12-04 23:54:29 +01:00
|
|
|
for name in names:
|
|
|
|
if _run_service_command('start', name):
|
|
|
|
win_msg(_("'%s' service started") % name)
|
|
|
|
else:
|
|
|
|
raise YunoHostError(1, _("Service starting failed for '%s'") % name)
|
|
|
|
|
|
|
|
|
|
|
|
def service_stop(names):
|
|
|
|
"""
|
|
|
|
Stop one or more services
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
name -- Services name to stop
|
|
|
|
|
|
|
|
"""
|
2013-12-17 00:26:27 +01:00
|
|
|
if isinstance(names, str):
|
|
|
|
names = [names]
|
2013-12-04 23:54:29 +01:00
|
|
|
for name in names:
|
|
|
|
if _run_service_command('stop', name):
|
|
|
|
win_msg(_("'%s' service stopped") % name)
|
|
|
|
else:
|
|
|
|
raise YunoHostError(1, _("Service stopping failed for '%s'") % name)
|
|
|
|
|
|
|
|
|
|
|
|
def service_enable(names):
|
|
|
|
"""
|
|
|
|
Enable one or more services
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
names -- Services name to enable
|
|
|
|
|
|
|
|
"""
|
2013-12-17 00:26:27 +01:00
|
|
|
if isinstance(names, str):
|
|
|
|
names = [names]
|
2013-12-04 23:54:29 +01:00
|
|
|
for name in names:
|
|
|
|
if _run_service_command('enable', name):
|
|
|
|
win_msg(_("'%s' service enabled") % name)
|
|
|
|
else:
|
|
|
|
raise YunoHostError(1, _("Service enabling failed for '%s'") % name)
|
|
|
|
|
|
|
|
|
|
|
|
def service_disable(names):
|
|
|
|
"""
|
|
|
|
Disable one or more services
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
names -- Services name to disable
|
|
|
|
|
|
|
|
"""
|
2013-12-17 00:26:27 +01:00
|
|
|
if isinstance(names, str):
|
|
|
|
names = [names]
|
2013-12-04 23:54:29 +01:00
|
|
|
for name in names:
|
|
|
|
if _run_service_command('disable', name):
|
|
|
|
win_msg(_("'%s' service disabled") % name)
|
|
|
|
else:
|
|
|
|
raise YunoHostError(1, _("Service disabling failed for '%s'") % name)
|
|
|
|
|
|
|
|
|
2013-12-17 00:26:27 +01:00
|
|
|
def service_status(names=[]):
|
2013-12-04 23:54:29 +01:00
|
|
|
"""
|
|
|
|
Show status information about one or more services (all by default)
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
names -- Services name to show
|
|
|
|
|
|
|
|
"""
|
|
|
|
services = _get_services()
|
|
|
|
check_names = True
|
|
|
|
result = {}
|
|
|
|
|
2013-12-17 00:26:27 +01:00
|
|
|
if isinstance(names, str):
|
|
|
|
names = [names]
|
|
|
|
elif len(names) == 0:
|
2013-12-04 23:54:29 +01:00
|
|
|
names = services.keys()
|
|
|
|
check_names = False
|
|
|
|
|
|
|
|
for name in names:
|
|
|
|
if check_names and name not in services.keys():
|
|
|
|
raise YunoHostError(1, _("Unknown service '%s'") % name)
|
|
|
|
|
|
|
|
status = None
|
|
|
|
if services[name]['status'] == 'service':
|
|
|
|
status = 'service %s status' % name
|
|
|
|
else:
|
|
|
|
status = str(services[name]['status'])
|
|
|
|
|
|
|
|
runlevel = 5
|
|
|
|
if 'runlevel' in services[name].keys():
|
|
|
|
runlevel = int(services[name]['runlevel'])
|
|
|
|
|
|
|
|
result[name] = { 'status': 'unknown', 'loaded': 'unknown' }
|
|
|
|
|
|
|
|
# Retrieve service status
|
|
|
|
try:
|
|
|
|
ret = subprocess.check_output(status.split(), stderr=subprocess.STDOUT)
|
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
# TODO: log error
|
2013-12-12 00:22:51 +01:00
|
|
|
if 'usage:' not in e.output.lower():
|
|
|
|
result[name]['status'] = _("inactive")
|
2013-12-04 23:54:29 +01:00
|
|
|
else:
|
|
|
|
result[name]['status'] = _("running")
|
|
|
|
|
|
|
|
# Retrieve service loading
|
|
|
|
rc_path = glob.glob("/etc/rc%d.d/S[0-9][0-9]%s" % (runlevel, name))
|
|
|
|
if len(rc_path) == 1 and os.path.islink(rc_path[0]):
|
|
|
|
result[name]['loaded'] = _("enabled")
|
|
|
|
elif os.path.isfile("/etc/init.d/%s" % name):
|
|
|
|
result[name]['loaded'] = _("disabled")
|
|
|
|
else:
|
|
|
|
result[name]['loaded'] = _("not-found")
|
|
|
|
|
2013-12-17 00:26:27 +01:00
|
|
|
if len(names) == 1:
|
|
|
|
return result[names[0]]
|
2013-12-04 23:54:29 +01:00
|
|
|
return result
|
|
|
|
|
|
|
|
|
2013-12-07 11:11:29 +01:00
|
|
|
def service_log(name, number=50):
|
2013-12-06 12:01:43 +01:00
|
|
|
"""
|
|
|
|
Log every log files of a service
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
name -- Services name to log
|
2013-12-07 11:11:29 +01:00
|
|
|
number -- Number of lines to display
|
2013-12-06 12:01:43 +01:00
|
|
|
|
|
|
|
"""
|
|
|
|
services = _get_services()
|
|
|
|
|
|
|
|
if name not in services.keys():
|
|
|
|
raise YunoHostError(1, _("Unknown service '%s'") % service)
|
2014-02-05 02:01:03 +01:00
|
|
|
|
2013-12-06 12:01:43 +01:00
|
|
|
if 'log' in services[name]:
|
|
|
|
log_list = services[name]['log']
|
|
|
|
result = {}
|
|
|
|
if not isinstance(log_list, list):
|
|
|
|
log_list = [log_list]
|
|
|
|
|
|
|
|
for log_path in log_list:
|
|
|
|
if os.path.isdir(log_path):
|
|
|
|
for log in [ f for f in os.listdir(log_path) if os.path.isfile(os.path.join(log_path, f)) and f[-4:] == '.log' ]:
|
2013-12-07 11:11:29 +01:00
|
|
|
result[os.path.join(log_path, log)] = _tail(os.path.join(log_path, log), int(number))
|
2013-12-06 12:01:43 +01:00
|
|
|
else:
|
2013-12-07 11:11:29 +01:00
|
|
|
result[log_path] = _tail(log_path, int(number))
|
2013-12-06 12:01:43 +01:00
|
|
|
else:
|
|
|
|
raise YunoHostError(1, _("Nothing to log for service '%s'") % name)
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2013-12-04 23:54:29 +01:00
|
|
|
def _run_service_command(action, service):
|
|
|
|
"""
|
|
|
|
Run services management command (start, stop, enable, disable)
|
|
|
|
|
|
|
|
Keyword argument:
|
|
|
|
service -- Service name
|
|
|
|
action -- Action to perform
|
|
|
|
|
|
|
|
"""
|
|
|
|
if service not in _get_services().keys():
|
|
|
|
raise YunoHostError(1, _("Unknown service '%s'") % service)
|
|
|
|
|
|
|
|
cmd = None
|
|
|
|
if action in ['start', 'stop']:
|
|
|
|
cmd = 'service %s %s' % (service, action)
|
|
|
|
elif action in ['enable', 'disable']:
|
|
|
|
arg = 'defaults' if action == 'enable' else 'remove'
|
|
|
|
cmd = 'update-rc.d %s %s' % (service, arg)
|
|
|
|
else:
|
|
|
|
raise YunoHostError(1, _("Unknown action '%s'") % service)
|
|
|
|
|
|
|
|
try:
|
|
|
|
ret = subprocess.check_output(cmd.split(), stderr=subprocess.STDOUT)
|
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
# TODO: log error instead
|
|
|
|
if os.isatty(1):
|
|
|
|
err = e.output.rstrip()
|
|
|
|
print(_("'%s' has returned:\n%s") % (' '.join(e.cmd), err))
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def _get_services():
|
|
|
|
"""
|
|
|
|
Get a dict of managed services with their parameters
|
|
|
|
|
|
|
|
"""
|
2013-12-05 15:58:41 +01:00
|
|
|
with open('/etc/yunohost/services.yml', 'r') as f:
|
2013-12-04 23:54:29 +01:00
|
|
|
services = yaml.load(f)
|
|
|
|
return services
|
2013-12-06 12:01:43 +01:00
|
|
|
|
|
|
|
|
|
|
|
def _tail(file, n, offset=None):
|
2013-12-06 12:05:34 +01:00
|
|
|
"""
|
|
|
|
Reads a n lines from f with an offset of offset lines. The return
|
2013-12-06 12:01:43 +01:00
|
|
|
value is a tuple in the form ``(lines, has_more)`` where `has_more` is
|
|
|
|
an indicator that is `True` if there are more lines in the file.
|
2013-12-06 12:05:34 +01:00
|
|
|
|
2013-12-06 12:01:43 +01:00
|
|
|
"""
|
|
|
|
avg_line_length = 74
|
|
|
|
to_read = n + (offset or 0)
|
|
|
|
|
2013-12-06 12:05:34 +01:00
|
|
|
try:
|
|
|
|
with open(file, 'r') as f:
|
|
|
|
while 1:
|
|
|
|
try:
|
|
|
|
f.seek(-(avg_line_length * to_read), 2)
|
|
|
|
except IOError:
|
|
|
|
# woops. apparently file is smaller than what we want
|
|
|
|
# to step back, go to the beginning instead
|
|
|
|
f.seek(0)
|
|
|
|
pos = f.tell()
|
|
|
|
lines = f.read().splitlines()
|
|
|
|
if len(lines) >= to_read or pos == 0:
|
2014-02-05 02:01:03 +01:00
|
|
|
return lines[-to_read:offset and -offset or None]
|
2013-12-06 12:05:34 +01:00
|
|
|
avg_line_length *= 1.3
|
|
|
|
|
|
|
|
except IOError: return []
|