]> git.llucax.com Git - software/pymin.git/blob - pymind
Shutdown logging properly when exiting pymind.
[software/pymin.git] / pymind
1 #!/usr/bin/env python
2 # vim: set encoding=utf-8 et sw=4 sts=4 :
3
4 import logging ; log = logging.getLogger('pymind')
5 # First of all, we need to setup the logging framework
6 # FIXME: this should go in a configuration file
7 logging.basicConfig(
8         level   = logging.DEBUG,
9         format  = '%(asctime)s %(name)-24s %(levelname)-8s %(message)s',
10         datefmt = '%a, %d %b %Y %H:%M:%S',
11 )
12
13 import os
14 import sys
15 from formencode import Invalid, validators as V
16
17 from pymin.config import OptionGroup, Option, ConfigOption, ListOption
18 from pymin.config import load_config, options
19 from pymin.config import ConfigError, MissingSectionHeaderError, ParsingError
20 from pymin.dispatcher import Handler
21 from pymin.pymindaemon import PyminDaemon
22 from pymin.service import load_service, LoadError
23
24 # exit status (1 is reserved for command-line errors)
25 EXIT_CONFIG_ERROR = 2
26 EXIT_NO_SERVICE   = 3
27
28 # default locations where to look for configuration files
29 # all found files will be processed, overriding the previous configuration
30 # files values.
31 config_file_paths = [
32     '/etc/pymin.ini',
33     '/etc/pymin/pymin.ini',
34     os.path.expanduser('~/.pymin.ini'),
35     os.path.expanduser('~/.pymin/pymin.ini'),
36 ]
37
38 # default locations where to look for service plug-ins
39 # search stops when a service plug-in is found
40 services_paths = [
41     os.path.expanduser('~/.pymin/services'),
42     '/usr/share/pymin/services',
43 ]
44
45 # default configuration variables
46 # these are useful variables to help the user writing the configuration file
47 config_defaults = {
48     'pymind-data-dir':   '/var/lib/pymin',
49     'pymind-pickle-dir': '/var/lib/pymin/pickle',
50     'pymind-config-dir': '/var/lib/pymin/config',
51 }
52
53 # Validator to check if is a valid Python identifier
54 PythonIdentifier = V.Regex(r'^[a-zA-Z_]\w*$')
55
56 options.init('pymind', 'Pymin daemon global options', [
57     Option('bind_addr', V.CIDR, 'a', default='127.0.0.1', metavar='ADDR',
58            help='Bind to IP ADDR'),
59     Option('bind_port', V.Int(min=1, max=65535), 'p', default=9999,
60            metavar='PORT', help="Bind to port PORT"),
61     ListOption('services', PythonIdentifier, 's', default=[],
62                metavar='SERVICE', help="manage service SERVICE"),
63     ListOption('services_dirs', V.String, 'd', default=[],
64                metavar='DIR', help="search for services in DIR"),
65     ConfigOption('config_file', 'c', metavar='FILE',
66                  help="load the configuration file FILE after the default "
67                       "configuration files"),
68     ConfigOption('replace_config_file', 'C', override=True, metavar='FILE',
69                  help="don't load the default configuration files, just FILE"),
70 ])
71
72
73 # FIXME: move to IpHandler or someplace else
74 def activate_ip_forward():
75     try:
76         f = file("/proc/sys/net/ipv4/ip_forward","w")
77         f.write("1")
78         f.close()
79     except (IOError, OSError), e:
80         log.warning("Can't set ip_forward: %s", e)
81
82
83 def die(status, msg, *args):
84     log.critical(msg, *args)
85     logging.shutdown()
86     sys.exit(status)
87
88 def get_config(paths, version, desc, add_options, defaults):
89     global config_file_paths
90     try:
91         (config, args) = load_config(paths, version, desc, add_options, defaults)
92     except ConfigError, e:
93         die(EXIT_CONFIG_ERROR, str(e))
94     except MissingSectionHeaderError, e:
95         dir(EXIT_CONFIG_ERROR, "%s:%s: missing section header near: %s",
96             e.filename, e.lineno, e.line)
97     except ParsingError, e:
98         for (lineno, line) in e.errors:
99             log.critical("%s:%s: invalid syntax near: %s", e.filename, lineno, line)
100         die(EXIT_CONFIG_ERROR, str(e.errors))
101     except Invalid, e:
102         die(EXIT_CONFIG_ERROR, str(e.unpack_errors()))
103     except LoadError, e:
104         die(EXIT_NO_SERVICE, "service '%s' not found (see option " \
105             "--services-dir)", e.service_name)
106     return (config, args)
107
108
109 class Services:
110     def __init__(self):
111         self.services = dict()
112     def add_config_options(self, config, args):
113         for service in config.services:
114             s = load_service(service, config.services_dirs)
115             s.setup_service(options, config)
116             self.services[service] = s
117
118 def build_root(config, args, services):
119     from pymin.dispatcher import Handler
120     class Root(Handler):
121         pass
122     # TODO check services dependencies
123     root = Root()
124     for name, service in services.items():
125         setattr(root, name, service.get_service(config))
126     return root
127
128
129 def main():
130     services = Services()
131     (config, args) = get_config(config_file_paths, '%prog 0.1',
132                                 'Router services administration daemon',
133                                 services.add_config_options, config_defaults)
134     root_handler = build_root(config, args, services.services)
135     activate_ip_forward()
136     PyminDaemon(root_handler, (config.bind_addr, config.bind_port)).run()
137     logging.shutdown()
138
139 if __name__ == '__main__':
140     main()
141