1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
# Copyright 2009 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
"""Start and stop the Mailman processes."""
__metaclass__ = type
__all__ = [
'start_mailman',
'stop_mailman',
]
import errno
import os
import signal
import subprocess
import sys
import canonical
from lp.services.mailman.config import configure_prefix
from lp.services.mailman.monkeypatches import monkey_patch
def mailmanctl(command, quiet=False, config=None, *additional_arguments):
"""Run mailmanctl command.
:param command: the command to use.
:param quiet: when this is true, no output will happen unless, an error
happens.
:param config: The CanonicalConfig object to take configuration from.
Defaults to the global one.
:param additional_arguments: additional command arguments to pass to the
mailmanctl program.
:raises RuntimeError: when quiet is True and the command failed.
"""
if config is None:
config = canonical.config.config
mailman_path = configure_prefix(config.mailman.build_prefix)
mailman_bin = os.path.join(mailman_path, 'bin')
args = ['./mailmanctl']
args.extend(additional_arguments)
args.append(command)
if quiet:
stdout = subprocess.PIPE
stderr = subprocess.STDOUT
else:
stdout = None
stderr = None
env = dict(os.environ)
env['LPCONFIG'] = config.instance_name
process = subprocess.Popen(
args, cwd=mailman_bin, stdout=stdout, stderr=stderr, env=env)
code = process.wait()
if code:
if quiet:
raise RuntimeError(
'mailmanctl %s failed: %d\n%s' % (
command, code, process.stdout.read()))
else:
print >> sys.stderr, 'mailmanctl %s failed: %d' % (command, code)
def stop_mailman(quiet=False, config=None):
"""Alias for mailmanctl('stop')."""
mailmanctl('stop', quiet, config)
# Further, if the Mailman master pid file was not removed, then the
# master watcher, and probably one of its queue runners, did not die.
# Kill it hard and clean up after it.
if config is None:
config = canonical.config.config
mailman_path = configure_prefix(config.mailman.build_prefix)
master_pid_path = os.path.join(mailman_path, 'data', 'master-qrunner.pid')
try:
master_pid_file = open(master_pid_path)
except IOError, error:
if error.errno == errno.ENOENT:
# It doesn't exist, so we're all done.
return
raise
try:
master_pid = int(master_pid_file.read().strip())
finally:
master_pid_file.close()
try:
# Kill the entire process group.
os.kill(master_pid, -signal.SIGKILL)
except OSError, error:
if error.errno == errno.ESRCH:
# The process does not exist. It could be a zombie that has yet
# to be waited on, but let's not worry about that.
return
raise
try:
os.remove(master_pid_path)
except OSError, error:
if error.errno != errno.ENOENT:
raise
lock_dir = os.path.join(mailman_path, 'locks')
for filename in os.listdir(lock_dir):
os.remove(os.path.join(lock_dir, filename))
def start_mailman(quiet=False, config=None):
"""Start the Mailman master qrunner.
The client of start_mailman() is responsible for ensuring that
stop_mailman() is called at the appropriate time.
:param quiet: when this is true, no output will happen unless, an error
happens.
:param config: The CanonicalConfig object to take configuration from.
Defaults to the global one.
:raises RuntimeException: when Mailman fails to start successfully.
"""
if config is None:
config = canonical.config.config
# We need the Mailman bin directory so we can run some of Mailman's
# command line scripts.
mailman_path = configure_prefix(config.mailman.build_prefix)
mailman_bin = os.path.join(mailman_path, 'bin')
# Monkey-patch the installed Mailman 2.1 tree.
monkey_patch(mailman_path, config)
# Start Mailman. Pass in the -s flag so that any stale master pid files
# will get deleted. "Stale" means the process that owned the pid no
# longer exists, so this can't hurt anything.
mailmanctl('start', quiet, config, '-s')
|