Passed
Push — master ( f5b357...dcdf4c )
by Lakshmi
03:24
created

ProcessSensorContainer.add_sensor()   A

Complexity

Conditions 2

Size

Total Lines 16

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 2
dl 0
loc 16
rs 9.4285
c 0
b 0
f 0
1
# Licensed to the StackStorm, Inc ('StackStorm') under one or more
2
# contributor license agreements.  See the NOTICE file distributed with
3
# this work for additional information regarding copyright ownership.
4
# The ASF licenses this file to You under the Apache License, Version 2.0
5
# (the "License"); you may not use this file except in compliance with
6
# the License.  You may obtain a copy of the License at
7
#
8
#     http://www.apache.org/licenses/LICENSE-2.0
9
#
10
# Unless required by applicable law or agreed to in writing, software
11
# distributed under the License is distributed on an "AS IS" BASIS,
12
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
# See the License for the specific language governing permissions and
14
# limitations under the License.
15
16
import os
17
import sys
18
import time
19
import json
20
import subprocess
21
22
from collections import defaultdict
23
24
import eventlet
25
from eventlet.support import greenlets as greenlet
26
from oslo_config import cfg
27
28
from st2common import log as logging
29
from st2common.constants.error_messages import PACK_VIRTUALENV_DOESNT_EXIST
30
from st2common.constants.system import API_URL_ENV_VARIABLE_NAME
31
from st2common.constants.system import AUTH_TOKEN_ENV_VARIABLE_NAME
32
from st2common.constants.triggers import (SENSOR_SPAWN_TRIGGER, SENSOR_EXIT_TRIGGER)
33
from st2common.constants.exit_codes import SUCCESS_EXIT_CODE
34
from st2common.constants.exit_codes import FAILURE_EXIT_CODE
35
from st2common.models.system.common import ResourceReference
36
from st2common.persistence.pack import Pack
37
from st2common.services.access import create_token
38
from st2common.transport.reactor import TriggerDispatcher
39
from st2common.util.api import get_full_public_api_url
40
from st2common.util.pack import get_pack_common_libs_path
41
from st2common.util.shell import on_parent_exit
42
from st2common.util.sandboxing import get_sandbox_python_path
43
from st2common.util.sandboxing import get_sandbox_python_binary_path
44
from st2common.util.sandboxing import get_sandbox_virtualenv_path
45
46
__all__ = [
47
    'ProcessSensorContainer'
48
]
49
50
LOG = logging.getLogger('st2reactor.process_sensor_container')
51
52
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
53
WRAPPER_SCRIPT_NAME = 'sensor_wrapper.py'
54
WRAPPER_SCRIPT_PATH = os.path.join(BASE_DIR, WRAPPER_SCRIPT_NAME)
55
56
# How many times to try to subsequently respawn a sensor after a non-zero exit before giving up
57
SENSOR_MAX_RESPAWN_COUNTS = 2
58
59
# How many seconds after the sensor has been started we should wait before considering sensor as
60
# being started and running successfully
61
SENSOR_SUCCESSFUL_START_THRESHOLD = 10
62
63
# How long to wait (in seconds) before respawning a dead process
64
SENSOR_RESPAWN_DELAY = 2.5
65
66
# How long to wait for process to exit after sending SIGTERM signal. If the process doesn't
67
# exit in this amount of seconds, SIGKILL signal will be sent to the process.
68
PROCESS_EXIT_TIMEOUT = 5
69
70
# TODO: Allow multiple instances of the same sensor with different configuration
71
# options - we need to update sensors for that and add "get_id" or similar
72
# method to the sensor class
73
74
75
class ProcessSensorContainer(object):
76
    """
77
    Sensor container which runs sensors in a separate process.
78
    """
79
80
    def __init__(self, sensors, poll_interval=5, dispatcher=None):
81
        """
82
        :param sensors: A list of sensor dicts.
83
        :type sensors: ``list`` of ``dict``
84
85
        :param poll_interval: How long to sleep between each poll for running / dead sensors.
86
        :type poll_interval: ``float``
87
        """
88
        self._poll_interval = poll_interval
89
90
        self._sensors = {}  # maps sensor_id -> sensor object
91
        self._processes = {}  # maps sensor_id -> sensor process
92
93
        if not dispatcher:
94
            dispatcher = TriggerDispatcher(LOG)
95
        self._dispatcher = dispatcher
96
97
        self._stopped = False
98
99
        sensors = sensors or []
100
        for sensor_obj in sensors:
101
            sensor_id = self._get_sensor_id(sensor=sensor_obj)
102
            self._sensors[sensor_id] = sensor_obj
103
104
        # Stores information needed for respawning dead sensors
105
        self._sensor_start_times = {}  # maps sensor_id -> sensor start time
106
        self._sensor_respawn_counts = defaultdict(int)  # maps sensor_id -> number of respawns
107
108
        # A list of all the instance variables which hold internal state information about a
109
        # particular_sensor
110
        # Note: We don't clear respawn counts since we want to track this through the whole life
111
        # cycle of the container manager
112
        self._internal_sensor_state_variables = [
113
            self._processes,
114
            self._sensors,
115
            self._sensor_start_times,
116
        ]
117
118
        self._enable_common_pack_libs = cfg.CONF.packs.enable_common_libs or False
119
120
    def run(self):
121
        self._run_all_sensors()
122
123
        try:
124
            while not self._stopped:
125
                # Poll for all running processes
126
                sensor_ids = self._sensors.keys()
127
128
                if len(sensor_ids) >= 1:
129
                    LOG.debug('%d active sensor(s)' % (len(sensor_ids)))
130
                    self._poll_sensors_for_results(sensor_ids)
131
                else:
132
                    LOG.debug('No active sensors')
133
134
                eventlet.sleep(self._poll_interval)
135
        except greenlet.GreenletExit:
136
            # This exception is thrown when sensor container manager
137
            # kills the thread which runs process container. Not sure
138
            # if this is the best thing to do.
139
            self._stopped = True
140
            return SUCCESS_EXIT_CODE
141
        except:
142
            LOG.exception('Container failed to run sensors.')
143
            self._stopped = True
144
            return FAILURE_EXIT_CODE
145
146
        self._stopped = True
147
        LOG.error('Process container quit. It shouldn\'t.')
148
        return SUCCESS_EXIT_CODE
149
150
    def _poll_sensors_for_results(self, sensor_ids):
151
        """
152
        Main loop which polls sensor for results and detects dead sensors.
153
        """
154
        for sensor_id in sensor_ids:
155
            now = int(time.time())
156
157
            process = self._processes[sensor_id]
158
            status = process.poll()
159
160
            if status is not None:
161
                # Dead process detected
162
                LOG.info('Process for sensor %s has exited with code %s', sensor_id, status)
163
164
                sensor = self._sensors[sensor_id]
165
                self._delete_sensor(sensor_id)
166
167
                self._dispatch_trigger_for_sensor_exit(sensor=sensor,
168
                                                       exit_code=status)
169
170
                # Try to respawn a dead process (maybe it was a simple failure which can be
171
                # resolved with a restart)
172
                eventlet.spawn_n(self._respawn_sensor, sensor_id=sensor_id, sensor=sensor,
173
                                 exit_code=status)
174
            else:
175
                sensor_start_time = self._sensor_start_times[sensor_id]
176
                sensor_respawn_count = self._sensor_respawn_counts[sensor_id]
177
                successfuly_started = (now - sensor_start_time) >= SENSOR_SUCCESSFUL_START_THRESHOLD
178
179
                if successfuly_started and sensor_respawn_count >= 1:
180
                    # Sensor has been successfully running more than threshold seconds, clear the
181
                    # respawn counter so we can try to restart the sensor if it dies later on
182
                    self._sensor_respawn_counts[sensor_id] = 0
183
184
    def running(self):
185
        return len(self._processes)
186
187
    def stopped(self):
188
        return self._stopped
189
190
    def shutdown(self, force=False):
191
        LOG.info('Container shutting down. Invoking cleanup on sensors.')
192
        self._stopped = True
193
194
        if force:
195
            exit_timeout = 0
196
        else:
197
            exit_timeout = PROCESS_EXIT_TIMEOUT
198
199
        sensor_ids = self._sensors.keys()
200
        for sensor_id in sensor_ids:
201
            self._stop_sensor_process(sensor_id=sensor_id, exit_timeout=exit_timeout)
202
203
        LOG.info('All sensors are shut down.')
204
205
        self._sensors = {}
206
        self._processes = {}
207
208
    def add_sensor(self, sensor):
209
        """
210
        Add a new sensor to the container.
211
212
        :type sensor: ``dict``
213
        """
214
        sensor_id = self._get_sensor_id(sensor=sensor)
215
216
        if sensor_id in self._sensors:
217
            LOG.warning('Sensor %s already exists and running.', sensor_id)
218
            return False
219
220
        self._spawn_sensor_process(sensor=sensor)
221
        LOG.debug('Sensor %s started.', sensor_id)
222
        self._sensors[sensor_id] = sensor
223
        return True
224
225
    def remove_sensor(self, sensor):
226
        """
227
        Remove an existing sensor from the container.
228
229
        :type sensor: ``dict``
230
        """
231
        sensor_id = self._get_sensor_id(sensor=sensor)
232
233
        if sensor_id not in self._sensors:
234
            LOG.warning('Sensor %s isn\'t running in this container.', sensor_id)
235
            return False
236
237
        self._stop_sensor_process(sensor_id=sensor_id)
238
        LOG.debug('Sensor %s stopped.', sensor_id)
239
        return True
240
241
    def _run_all_sensors(self):
242
        sensor_ids = self._sensors.keys()
243
244
        for sensor_id in sensor_ids:
245
            sensor_obj = self._sensors[sensor_id]
246
            LOG.info('Running sensor %s', sensor_id)
247
248
            try:
249
                self._spawn_sensor_process(sensor=sensor_obj)
250
            except Exception as e:
251
                LOG.warning(e.message, exc_info=True)
252
253
                # Disable sensor which we are unable to start
254
                del self._sensors[sensor_id]
255
                continue
256
257
            LOG.info('Sensor %s started' % sensor_id)
258
259
    def _spawn_sensor_process(self, sensor):
260
        """
261
        Spawn a new process for the provided sensor.
262
263
        New process uses isolated Python binary from a virtual environment
264
        belonging to the sensor pack.
265
        """
266
        sensor_id = self._get_sensor_id(sensor=sensor)
267
        pack_ref = sensor['pack']
268
        pack_db = Pack.get_by_ref(pack_ref)
269
270
        virtualenv_path = get_sandbox_virtualenv_path(pack=pack_ref)
271
        python_path = get_sandbox_python_binary_path(pack=pack_ref)
272
273
        if virtualenv_path and not os.path.isdir(virtualenv_path):
274
            format_values = {'pack': sensor['pack'], 'virtualenv_path': virtualenv_path}
275
            msg = PACK_VIRTUALENV_DOESNT_EXIST % format_values
276
            raise Exception(msg)
277
278
        trigger_type_refs = sensor['trigger_types'] or []
279
        trigger_type_refs = ','.join(trigger_type_refs)
280
281
        parent_args = json.dumps(sys.argv[1:])
282
283
        args = [
284
            python_path,
285
            WRAPPER_SCRIPT_PATH,
286
            '--pack=%s' % (sensor['pack']),
287
            '--file-path=%s' % (sensor['file_path']),
288
            '--class-name=%s' % (sensor['class_name']),
289
            '--trigger-type-refs=%s' % (trigger_type_refs),
290
            '--parent-args=%s' % (parent_args)
291
        ]
292
293
        if sensor['poll_interval']:
294
            args.append('--poll-interval=%s' % (sensor['poll_interval']))
295
296
        sandbox_python_path = get_sandbox_python_path(inherit_from_parent=True,
297
                                                      inherit_parent_virtualenv=True)
298
        pack_common_libs_path = get_pack_common_libs_path(pack_db=pack_db)
299
300
        env = os.environ.copy()
301
302
        if self._enable_common_pack_libs and pack_common_libs_path:
303
            env['PYTHONPATH'] = pack_common_libs_path + ':' + sandbox_python_path
304
        else:
305
            env['PYTHONPATH'] = sandbox_python_path
306
307
        # Include full api URL and API token specific to that sensor
308
        ttl = cfg.CONF.auth.service_token_ttl
309
        metadata = {
310
            'service': 'sensors_container',
311
            'sensor_path': sensor['file_path'],
312
            'sensor_class': sensor['class_name']
313
        }
314
        temporary_token = create_token(username='sensors_container', ttl=ttl, metadata=metadata,
315
                                       service=True)
316
317
        env[API_URL_ENV_VARIABLE_NAME] = get_full_public_api_url()
318
        env[AUTH_TOKEN_ENV_VARIABLE_NAME] = temporary_token.token
319
320
        # TODO 1: Purge temporary token when service stops or sensor process dies
321
        # TODO 2: Store metadata (wrapper process id) with the token and delete
322
        # tokens for old, dead processes on startup
323
        cmd = ' '.join(args)
324
        LOG.debug('Running sensor subprocess (cmd="%s")', cmd)
325
326
        # TODO: Intercept stdout and stderr for aggregated logging purposes
327
        try:
328
            process = subprocess.Popen(args=args, stdin=None, stdout=None,
329
                                       stderr=None, shell=False, env=env,
330
                                       preexec_fn=on_parent_exit('SIGTERM'))
331
        except Exception as e:
332
            cmd = ' '.join(args)
333
            message = ('Failed to spawn process for sensor %s ("%s"): %s' %
334
                       (sensor_id, cmd, str(e)))
335
            raise Exception(message)
336
337
        self._processes[sensor_id] = process
338
        self._sensors[sensor_id] = sensor
339
        self._sensor_start_times[sensor_id] = int(time.time())
340
341
        self._dispatch_trigger_for_sensor_spawn(sensor=sensor, process=process, cmd=cmd)
342
343
        return process
344
345
    def _stop_sensor_process(self, sensor_id, exit_timeout=PROCESS_EXIT_TIMEOUT):
346
        """
347
        Stop a sensor process for the provided sensor.
348
349
        :param sensor_id: Sensor ID.
350
        :type sensor_id: ``str``
351
352
        :param exit_timeout: How long to wait for process to exit after
353
                             sending SIGTERM signal. If the process doesn't
354
                             exit in this amount of seconds, SIGKILL signal
355
                             will be sent to the process.
356
        :type exit__timeout: ``int``
357
        """
358
        process = self._processes[sensor_id]
359
360
        # Delete sensor before terminating process so that it will not be
361
        # respawned during termination
362
        self._delete_sensor(sensor_id)
363
364
        # Terminate the process and wait for up to stop_timeout seconds for the
365
        # process to exit
366
        process.terminate()
367
368
        timeout = 0
369
        sleep_delay = 1
370
        while timeout < exit_timeout:
371
            status = process.poll()
372
373
            if status is not None:
374
                # Process has exited
375
                break
376
377
            timeout += sleep_delay
378
            time.sleep(sleep_delay)
379
380
        if status is None:
381
            # Process hasn't exited yet, forcefully kill it
382
            process.kill()
383
384
    def _respawn_sensor(self, sensor_id, sensor, exit_code):
385
        """
386
        Method for respawning a sensor which died with a non-zero exit code.
387
        """
388
        extra = {'sensor_id': sensor_id, 'sensor': sensor}
389
390
        if self._stopped:
391
            LOG.debug('Stopped, not respawning a dead sensor', extra=extra)
392
            return
393
394
        should_respawn = self._should_respawn_sensor(sensor_id=sensor_id, sensor=sensor,
395
                                                     exit_code=exit_code)
396
397
        if not should_respawn:
398
            LOG.debug('Not respawning a dead sensor', extra=extra)
399
            return
400
401
        LOG.debug('Respawning dead sensor', extra=extra)
402
403
        self._sensor_respawn_counts[sensor_id] += 1
404
        sleep_delay = (SENSOR_RESPAWN_DELAY * self._sensor_respawn_counts[sensor_id])
405
        eventlet.sleep(sleep_delay)
406
407
        try:
408
            self._spawn_sensor_process(sensor=sensor)
409
        except Exception as e:
410
            LOG.warning(e.message, exc_info=True)
411
412
            # Disable sensor which we are unable to start
413
            del self._sensors[sensor_id]
414
415
    def _should_respawn_sensor(self, sensor_id, sensor, exit_code):
416
        """
417
        Return True if the provided sensor should be respawned, False otherwise.
418
        """
419
        if exit_code == 0:
420
            # We only try to respawn sensors which exited with non-zero status code
421
            return False
422
423
        respawn_count = self._sensor_respawn_counts[sensor_id]
424
        if respawn_count >= SENSOR_MAX_RESPAWN_COUNTS:
425
            LOG.debug('Sensor has already been respawned max times, giving up')
426
            return False
427
428
        return True
429
430
    def _get_sensor_id(self, sensor):
431
        """
432
        Return unique identifier for the provider sensor dict.
433
434
        :type sensor: ``dict``
435
        """
436
        sensor_id = sensor['ref']
437
        return sensor_id
438
439
    def _dispatch_trigger_for_sensor_spawn(self, sensor, process, cmd):
440
        trigger = ResourceReference.to_string_reference(
441
            name=SENSOR_SPAWN_TRIGGER['name'],
442
            pack=SENSOR_SPAWN_TRIGGER['pack'])
443
        now = int(time.time())
444
        payload = {
445
            'id': sensor['class_name'],
446
            'timestamp': now,
447
            'pid': process.pid,
448
            'cmd': cmd
449
        }
450
        self._dispatcher.dispatch(trigger, payload=payload)
451
452
    def _dispatch_trigger_for_sensor_exit(self, sensor, exit_code):
453
        trigger = ResourceReference.to_string_reference(
454
            name=SENSOR_EXIT_TRIGGER['name'],
455
            pack=SENSOR_EXIT_TRIGGER['pack'])
456
        now = int(time.time())
457
        payload = {
458
            'id': sensor['class_name'],
459
            'timestamp': now,
460
            'exit_code': exit_code
461
        }
462
        self._dispatcher.dispatch(trigger, payload=payload)
463
464
    def _delete_sensor(self, sensor_id):
465
        """
466
        Delete / reset all the internal state about a particular sensor.
467
        """
468
        for var in self._internal_sensor_state_variables:
469
            if sensor_id in var:
470
                del var[sensor_id]
471