~launchpad-pqm/launchpad/devel

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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#!/usr/bin/python -S
#
# Copyright 2009 Canonical Ltd.  This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).

"""Startup and shutdown slon processes.

On production and staging we probably want to use the standard
/etc/init.d/slony1 script instead of this tool.
"""

import _pythonpath

import os.path
import subprocess
import sys
from optparse import OptionParser

from canonical.config import config
from canonical.database.postgresql import ConnectionString
from canonical.database.sqlbase import connect
from canonical.launchpad.scripts import logger, logger_options
import replication.helpers

__metaclass__ = type
__all__ = []


def main():
    parser = OptionParser(
        "Usage: %prog [options] "
        "[start [nickname connection_string] | stop [nickname]]")
    parser.add_option(
        '-l', '--lag', default=None, dest="lag", metavar='PGINTERVAL',
        help="Lag events by PGINTERVAL, such as '10 seconds' or '2 minutes'")
    logger_options(parser)
    options, args = parser.parse_args()

    if len(args) == 0:
        parser.error("No command given.")

    command = args[0]

    if command not in ['start', 'stop']:
        parser.error("Unknown command %s." % command)

    if len(args) == 1:
        explicit = None

    elif command == 'start':
        if len(args) == 2:
            parser.error(
                "nickname and connection_string required for %s command"
                % command)
        elif len(args) == 3:
            explicit = replication.helpers.Node(
                None, args[1], args[2], None)
        else:
            parser.error("Too many arguments.")

    elif command == 'stop':
        if len(args) in (2, 3):
            explicit = replication.helpers.Node(
                None, args[1], None, None)
        else:
            parser.error("Too many arguments.")

    else:
        parser.error("Unknown command %s." % command)

    log = logger(options)

    if explicit is not None:
        nodes = [explicit]
    else:
        nodes = replication.helpers.get_all_cluster_nodes(connect('slony'))

    if command == 'start':
        return start(log, nodes, options.lag)
    else:
        return stop(log, nodes)


def get_pidfile(nickname):
    return os.path.join(
        config.canonical.pid_dir, 'lpslon_%s_%s.pid' % (
        nickname, config.instance_name))


def get_logfile(nickname):
    logdir = config.database.replication_logdir
    if not os.path.isabs(logdir):
        logdir = os.path.normpath(os.path.join(config.root, logdir))
    return os.path.join(
        logdir, 'lpslon_%s_%s.log' % (nickname, config.instance_name))


def start(log, nodes, lag=None):
    for node in nodes:
        pidfile = get_pidfile(node.nickname)
        logfile = get_logfile(node.nickname)

        log.info("Starting %s slon daemon." % node.nickname)
        log.debug("Logging to %s" % logfile)
        log.debug("PID file %s" % pidfile)
        # Hard code suitable command line arguments for development.
        slon_args = "-d 2 -s 500 -t 2500"
        if lag is not None:
            slon_args = "%s -l '%s'" % (slon_args, lag)
        cmd = [
            "/sbin/start-stop-daemon",
            "--start",
            "--background",
            "--pidfile", pidfile,
            "--oknodo",
            "--exec", "/usr/bin/slon",
            "--startas", "/bin/sh",
            "--", "-c",
            "slon -p %s %s %s '%s' >> %s" % (
                pidfile, slon_args, replication.helpers.CLUSTERNAME,
                node.connection_string, logfile)]
        log.debug("Running %s" % repr(cmd))
        return_code = subprocess.call(cmd)
        if return_code != 0:
            log.fatal("Failed. Return code %s" % return_code)
            return return_code

    return 0


def stop(log, nodes):
    for node in nodes:
        pidfile = get_pidfile(node.nickname)
        logfile = get_logfile(node.nickname)

        if not os.path.exists(pidfile):
            log.info(
                "%s slon daemon not running. Doing nothing."
                % node.nickname)
            continue
        log.info("Stopping %s slon daemon." % node.nickname)
        log.debug("PID file %s" % pidfile)
        cmd = [
            "/sbin/start-stop-daemon",
            "--stop",
            "--pidfile", pidfile,
            "--oknodo"]
        log.debug("Running %s" % repr(cmd))
        return_code = subprocess.call(cmd)
        if return_code != 0:
            log.fatal("Failed. Return code %s" % return_code)
            return return_code

    return 0


if __name__ == '__main__':
    sys.exit(main())