koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 1 | import os |
| 2 | import sys |
| 3 | import time |
| 4 | import json |
koder aka kdanilov | e06762a | 2015-03-22 23:32:09 +0200 | [diff] [blame] | 5 | import glob |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 6 | import signal |
| 7 | import os.path |
| 8 | import argparse |
| 9 | |
koder aka kdanilov | cff7b2e | 2015-04-18 20:48:15 +0300 | [diff] [blame] | 10 | from .sensors.utils import SensorInfo |
| 11 | from .daemonize import Daemonize |
| 12 | from .discover import all_sensors |
| 13 | from .protocol import create_protocol |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 14 | |
| 15 | |
koder aka kdanilov | e06762a | 2015-03-22 23:32:09 +0200 | [diff] [blame] | 16 | # load all sensors |
koder aka kdanilov | cff7b2e | 2015-04-18 20:48:15 +0300 | [diff] [blame] | 17 | from . import sensors |
koder aka kdanilov | e06762a | 2015-03-22 23:32:09 +0200 | [diff] [blame] | 18 | sensors_dir = os.path.dirname(sensors.__file__) |
| 19 | for fname in glob.glob(os.path.join(sensors_dir, "*.py")): |
| 20 | mod_name = os.path.basename(fname[:-3]) |
koder aka kdanilov | cff7b2e | 2015-04-18 20:48:15 +0300 | [diff] [blame] | 21 | __import__("sensors.sensors." + mod_name) |
koder aka kdanilov | e06762a | 2015-03-22 23:32:09 +0200 | [diff] [blame] | 22 | |
| 23 | |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 24 | def get_values(required_sensors): |
| 25 | result = {} |
| 26 | for sensor_name, params in required_sensors: |
| 27 | if sensor_name in all_sensors: |
| 28 | result.update(all_sensors[sensor_name](**params)) |
| 29 | else: |
| 30 | msg = "Sensor {0!r} isn't available".format(sensor_name) |
| 31 | raise ValueError(msg) |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 32 | return result |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 33 | |
| 34 | |
| 35 | def parse_args(args): |
| 36 | parser = argparse.ArgumentParser() |
| 37 | parser.add_argument('-d', '--daemon', |
| 38 | choices=('start', 'stop', 'status'), |
| 39 | default=None) |
| 40 | |
| 41 | parser.add_argument('-u', '--url', default='stdout://') |
| 42 | parser.add_argument('-t', '--timeout', type=float, default=1) |
koder aka kdanilov | 8097a4e | 2015-03-18 11:07:35 +0200 | [diff] [blame] | 43 | parser.add_argument('-l', '--list-sensors', action='store_true') |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 44 | parser.add_argument('sensors_config', type=argparse.FileType('r'), |
| 45 | default=None, nargs='?') |
| 46 | return parser.parse_args(args[1:]) |
| 47 | |
| 48 | |
| 49 | def daemon_main(required_sensors, opts): |
koder aka kdanilov | e87ae65 | 2015-04-20 02:14:35 +0300 | [diff] [blame] | 50 | try: |
| 51 | source_id = str(required_sensors.pop('source_id')) |
| 52 | except KeyError: |
| 53 | source_id = None |
koder aka kdanilov | 168f609 | 2015-04-19 02:33:38 +0300 | [diff] [blame] | 54 | |
Alyona Kiseleva | 7f6de4f | 2015-04-21 01:04:20 +0300 | [diff] [blame] | 55 | sender = create_protocol(opts.url) |
| 56 | prev = {} |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 57 | next_data_record_time = time.time() |
Alyona Kiseleva | 7f6de4f | 2015-04-21 01:04:20 +0300 | [diff] [blame] | 58 | |
koder aka kdanilov | e87ae65 | 2015-04-20 02:14:35 +0300 | [diff] [blame] | 59 | while True: |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 60 | real_time = int(time.time()) |
| 61 | |
| 62 | if real_time < int(next_data_record_time): |
| 63 | if int(next_data_record_time) - real_time > 2: |
| 64 | print "Error: sleep too small portion!!" |
| 65 | report_time = int(next_data_record_time) |
| 66 | elif real_time > int(next_data_record_time): |
| 67 | if real_time - int(next_data_record_time) > 2: |
| 68 | report_time = real_time |
| 69 | else: |
| 70 | report_time = int(next_data_record_time) |
| 71 | else: |
| 72 | report_time = real_time |
| 73 | |
| 74 | data = get_values(required_sensors.items()) |
| 75 | curr = {'time': SensorInfo(report_time, True)} |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 76 | for name, val in data.items(): |
| 77 | if val.is_accumulated: |
| 78 | if name in prev: |
| 79 | curr[name] = SensorInfo(val.value - prev[name], True) |
| 80 | prev[name] = val.value |
| 81 | else: |
| 82 | curr[name] = SensorInfo(val.value, False) |
koder aka kdanilov | 168f609 | 2015-04-19 02:33:38 +0300 | [diff] [blame] | 83 | |
| 84 | if source_id is not None: |
| 85 | curr['source_id'] = source_id |
| 86 | |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 87 | print report_time, int((report_time - time.time()) * 10) * 0.1 |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 88 | sender.send(curr) |
Alyona Kiseleva | 7f6de4f | 2015-04-21 01:04:20 +0300 | [diff] [blame] | 89 | |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 90 | next_data_record_time = report_time + opts.timeout + 0.5 |
| 91 | time.sleep(next_data_record_time - time.time()) |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 92 | |
| 93 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 94 | def pid_running(pid): |
| 95 | return os.path.exists("/proc/" + str(pid)) |
| 96 | |
| 97 | |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 98 | def main(argv): |
| 99 | opts = parse_args(argv) |
| 100 | |
koder aka kdanilov | 8097a4e | 2015-03-18 11:07:35 +0200 | [diff] [blame] | 101 | if opts.list_sensors: |
koder aka kdanilov | e06762a | 2015-03-22 23:32:09 +0200 | [diff] [blame] | 102 | print "\n".join(sorted(all_sensors)) |
koder aka kdanilov | 8097a4e | 2015-03-18 11:07:35 +0200 | [diff] [blame] | 103 | return 0 |
| 104 | |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 105 | if opts.daemon is not None: |
| 106 | pid_file = "/tmp/sensors.pid" |
| 107 | if opts.daemon == 'start': |
| 108 | required_sensors = json.loads(opts.sensors_config.read()) |
| 109 | |
| 110 | def root_func(): |
| 111 | daemon_main(required_sensors, opts) |
| 112 | |
| 113 | daemon = Daemonize(app="perfcollect_app", |
| 114 | pid=pid_file, |
| 115 | action=root_func) |
| 116 | daemon.start() |
| 117 | elif opts.daemon == 'stop': |
| 118 | if os.path.isfile(pid_file): |
| 119 | pid = int(open(pid_file).read()) |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 120 | if pid_running(pid): |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 121 | os.kill(pid, signal.SIGTERM) |
| 122 | |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 123 | time.sleep(0.5) |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 124 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 125 | if pid_running(pid): |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 126 | os.kill(pid, signal.SIGKILL) |
| 127 | |
koder aka kdanilov | 57ce4db | 2015-04-25 21:25:51 +0300 | [diff] [blame] | 128 | time.sleep(0.5) |
| 129 | |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 130 | if os.path.isfile(pid_file): |
| 131 | os.unlink(pid_file) |
| 132 | elif opts.daemon == 'status': |
| 133 | if os.path.isfile(pid_file): |
| 134 | pid = int(open(pid_file).read()) |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 135 | if pid_running(pid): |
koder aka kdanilov | dda86d3 | 2015-03-16 11:20:04 +0200 | [diff] [blame] | 136 | print "running" |
| 137 | return |
| 138 | print "stopped" |
| 139 | else: |
| 140 | raise ValueError("Unknown daemon operation {}".format(opts.daemon)) |
| 141 | else: |
| 142 | required_sensors = json.loads(opts.sensors_config.read()) |
| 143 | daemon_main(required_sensors, opts) |
| 144 | return 0 |
| 145 | |
| 146 | if __name__ == "__main__": |
| 147 | exit(main(sys.argv)) |