ac2550 | 5082fd7 | 2018-03-20 12:35:48 +0100 | [diff] [blame] | 1 | #!/usr/bin/env python3 |
| 2 | import os |
| 3 | import json |
| 4 | import copy |
| 5 | import random |
| 6 | import requests |
| 7 | import uuid |
| 8 | import time |
| 9 | from datetime import datetime |
| 10 | |
| 11 | def luck(n=2): |
| 12 | """ gives 1 chance out of n (default: 2) to return True """ |
| 13 | assert n > 1 |
| 14 | return bool(random.randint(0, n-1)) |
| 15 | def now_dmaap_timestamp(): |
| 16 | return str(datetime.now().timestamp()).replace(".","")[:13] |
| 17 | def now_notification_time(): |
| 18 | return datetime.now().strftime("%Y-%m-%d %H:%M:%S.%f+00:00") |
| 19 | |
| 20 | CONTROL_LOOP_NAMES = [ |
| 21 | 'ClosedLoop-vUSP-SIG-d925ed73-8231-4d02-9545-db4e101f88f8', |
| 22 | 'ClosedLoop-vUSP-SIG-37b1c91e-fd6b-4abd-af15-771902d5fdb1', |
| 23 | 'ClosedLoop-vUSP-SIG-c2597657-7113-4efb-a1f9-397a7a24c3e1', |
| 24 | 'ClosedLoop-vUSP-SIG-a11318ba-4c61-46b8-a9da-bd40bec11d45', |
| 25 | 'ClosedLoop-vUSP-SIG-5321c558-2254-4efb-ac24-99dd54edc94f', |
| 26 | ] |
| 27 | |
| 28 | TEMPLATES = { |
| 29 | 'event_abated' :'event_abated.json', |
| 30 | 'event_onset' :'event_onset.json', |
| 31 | 'notification_active' :'notification_active.json', |
| 32 | 'notification_final_failed' :'notification_final_failed.json', |
| 33 | 'notification_final_open' :'notification_final_open.json', |
| 34 | 'notification_final_success' :'notification_final_success.json', |
| 35 | 'notification_operation_failure' :'notification_operation_failure.json', |
| 36 | 'notification_operation' :'notification_operation.json', |
| 37 | 'notification_operation_success' :'notification_operation_success.json', |
| 38 | 'notification_rejected_disabled' :'notification_rejected_disabled.json', |
| 39 | 'notification_rejected_missing' :'notification_rejected_missing.json', |
| 40 | } |
| 41 | |
| 42 | for key in TEMPLATES: |
| 43 | with open(TEMPLATES[key]) as f: |
| 44 | content = f.read() |
| 45 | TEMPLATES[key] = json.loads(content) |
| 46 | |
| 47 | |
| 48 | class DMaaPMessage(dict): |
| 49 | |
| 50 | dmaap_host_url = "http://dmaap.host.url:9200/" |
| 51 | dmaap_username = None |
| 52 | dmaap_password = None |
| 53 | |
| 54 | @classmethod |
| 55 | def from_template(cls, tmpl, **kwargs): |
| 56 | obj = cls() |
| 57 | obj.update(copy.deepcopy(TEMPLATES[tmpl])) |
| 58 | for keys,value in kwargs.items(): |
| 59 | current_node = obj |
| 60 | keys = keys.split(".") |
| 61 | key = keys[0] |
| 62 | for i in range(len(keys) - 1): |
| 63 | current_node = current_node[keys[i]] |
| 64 | key = keys[i] |
| 65 | current_node[key] = value |
| 66 | return obj |
| 67 | |
| 68 | def publish(self, topic): |
| 69 | url = "%s/events/%s" % (self.dmaap_host_url, topic) |
| 70 | auth = None |
| 71 | if self.dmaap_username and self.dmaap_password: |
| 72 | auth = (self.dmaap_username, self.dmaap_password) |
| 73 | response = requests.post(url, data=json.dumps(self), auth=auth) |
| 74 | return response.status_code |
| 75 | |
| 76 | class Event(DMaaPMessage): |
| 77 | |
| 78 | topic = "DCAE-CL-EVENT" |
| 79 | |
| 80 | @staticmethod |
| 81 | def abated(**kwargs): |
| 82 | return Event.from_template('event_abated', **kwargs) |
| 83 | |
| 84 | @staticmethod |
| 85 | def onset(**kwargs): |
| 86 | return Event.from_template('event_onset', **kwargs) |
| 87 | |
| 88 | def publish(self): |
| 89 | return super().publish(self.topic) |
| 90 | |
| 91 | |
| 92 | class Notification(DMaaPMessage): |
| 93 | |
| 94 | topic = "POLICY-CL-MGT" |
| 95 | |
| 96 | @classmethod |
| 97 | def from_template(cls, tmpl, **kwargs): |
| 98 | kwargs['notificationTime'] = now_notification_time() |
| 99 | return super().from_template(tmpl, **kwargs) |
| 100 | |
| 101 | @staticmethod |
| 102 | def active(**kwargs): |
| 103 | return Notification.from_template('notification_active', **kwargs) |
| 104 | |
| 105 | @staticmethod |
| 106 | def final(**kwargs): |
| 107 | class FinalNotification(Notification): |
| 108 | @staticmethod |
| 109 | def success(**kwargs): |
| 110 | return FinalNotification.from_template('notification_final_success', **kwargs) |
| 111 | @staticmethod |
| 112 | def failed(**kwargs): |
| 113 | return FinalNotification.from_template('notification_final_failed', **kwargs) |
| 114 | @staticmethod |
| 115 | def open(**kwargs): |
| 116 | return FinalNotification.from_template('notification_final_open', **kwargs) |
| 117 | return FinalNotification |
| 118 | |
| 119 | @staticmethod |
| 120 | def operation(**kwargs): |
| 121 | class OperationNotification(Notification): |
| 122 | @staticmethod |
| 123 | def success(**kwargs): |
| 124 | return OperationNotification.from_template('notification_operation_success', **kwargs) |
| 125 | @staticmethod |
| 126 | def failure(**kwargs): |
| 127 | return OperationNotification.from_template('notification_operation_failure', **kwargs) |
| 128 | return OperationNotification.from_template('notification_operation', **kwargs) |
| 129 | |
| 130 | @staticmethod |
| 131 | def rejected(**kwargs): |
| 132 | class RejectedNotification(Notification): |
| 133 | @staticmethod |
| 134 | def disabled(**kwargs): |
| 135 | return RejectedNotification.from_template('notification_rejected_disabled', **kwargs) |
| 136 | @staticmethod |
| 137 | def missing_fields(**kwargs): |
| 138 | return RejectedNotification.from_template('notification_rejected_missing', **kwargs) |
| 139 | |
| 140 | return RejectedNotification |
| 141 | |
| 142 | def publish(self): |
| 143 | return super().publish(self.topic) |
| 144 | |
| 145 | |
| 146 | |
| 147 | class CLStatus(object): |
| 148 | |
| 149 | def __init__(self, dmaap_url=None, |
| 150 | missing=None, disabled=None, op_failure=None): |
| 151 | self._stopped = False |
| 152 | def maybe(thing): |
| 153 | if thing is None: |
| 154 | thing = not luck(10) |
| 155 | return thing |
| 156 | self._missing = maybe(missing) |
| 157 | self._disabled = maybe(disabled) |
| 158 | self._op_failure = maybe(op_failure) |
| 159 | self._config = dict( |
| 160 | requestID=str(uuid.uuid4()), |
| 161 | closedLoopControlName=CONTROL_LOOP_NAMES[random.randint(0, len(CONTROL_LOOP_NAMES) - 1)] |
| 162 | ) |
| 163 | |
| 164 | def __iter__(self): |
| 165 | return next(self) |
| 166 | |
| 167 | def __next__(self): |
| 168 | if self._stopped: |
| 169 | raise StopIteration() |
| 170 | config = self._config |
| 171 | config.update(dict(closedLoopAlarmStart=now_dmaap_timestamp())) |
| 172 | yield Event.onset(**config) |
| 173 | if self._missing: |
| 174 | self._stopped = True |
| 175 | yield Notification.rejected().missing_fields(**config) |
| 176 | raise StopIteration() |
| 177 | elif self._disabled: |
| 178 | self._stopped = True |
| 179 | yield Notification.rejected().disabled(**config) |
| 180 | raise StopIteration() |
| 181 | |
| 182 | yield Notification.active(**config) |
| 183 | yield Notification.operation(**config) |
| 184 | |
| 185 | config['closedLoopAlarmEnd'] = now_dmaap_timestamp() |
| 186 | if self._op_failure: |
| 187 | yield Notification.operation().failure(**config) |
| 188 | self._stopped = True |
| 189 | yield Notification.final().failed(**config) |
| 190 | else: |
| 191 | yield Notification.operation().success(**config) |
| 192 | yield Event.abated(**config) |
| 193 | self._stopped = True |
| 194 | yield Notification.final().success(**config) |
| 195 | raise StopIteration() |
| 196 | |
ac2550 | 69a7eff | 2018-03-27 14:00:27 +0200 | [diff] [blame] | 197 | def print_usage(): |
| 198 | print(""" |
| 199 | ./ds_mocker.py <DMAAP_URL> <EVENT_TOPIC> [NOTIFICATION_TOPIC [REQUEST_TOPIC]] |
| 200 | """) |
| 201 | exit() |
| 202 | |
| 203 | def push(test_datas): |
| 204 | for current_i, status in enumerate(test_datas): |
| 205 | time.sleep(random.randint(0,6)) |
| 206 | for s in status: |
| 207 | # print(s) |
| 208 | status_code = s.publish() |
| 209 | if status_code != 200: |
| 210 | print("Error when publishing : status_code={}".format(status_code)) |
| 211 | exit(1) |
| 212 | time.sleep(random.randint(0,3)) |
| 213 | print("%03d,missing:%5s,disabled:%5s,op_failure:%5s - %s" % (current_i, status._missing, status._disabled, status._op_failure, status._config)) |
| 214 | |
| 215 | |
| 216 | def generate_dataset_1(): |
| 217 | test_datas = [CLStatus(missing=False, disabled=False, op_failure=False) for i in range(45)] \ |
ac2550 | 5082fd7 | 2018-03-20 12:35:48 +0100 | [diff] [blame] | 218 | + [CLStatus(missing=True, disabled=False, op_failure=False) for i in range(5)] \ |
| 219 | + [CLStatus(missing=False, disabled=True, op_failure=False) for i in range(6)] \ |
| 220 | + [CLStatus(missing=False, disabled=False, op_failure=True) for i in range(7)] |
ac2550 | 69a7eff | 2018-03-27 14:00:27 +0200 | [diff] [blame] | 221 | random.shuffle(test_datas) |
| 222 | return test_datas |
ac2550 | 5082fd7 | 2018-03-20 12:35:48 +0100 | [diff] [blame] | 223 | |
ac2550 | 69a7eff | 2018-03-27 14:00:27 +0200 | [diff] [blame] | 224 | |
| 225 | if __name__ == "__main__": |
| 226 | import sys |
| 227 | if len(sys.argv) < 3: |
| 228 | print_usage() |
| 229 | |
| 230 | DMaaPMessage.dmaap_host_url = sys.argv[1] |
| 231 | Event.topic = sys.argv[2] |
| 232 | Notification.topic = len(sys.argv) > 3 and sys.argv[3] or sys.argv[2] |
| 233 | # Request.topic = len(sys.argv) > 4 or Notification.topic |
| 234 | push(generate_dataset_1()) |