|  |  |  | import math
 | 
					
						
							|  |  |  | import os
 | 
					
						
							|  |  |  | from enum import IntEnum
 | 
					
						
							|  |  |  | from typing import Dict, Union, Callable, List, Optional
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from cereal import log, car
 | 
					
						
							|  |  |  | import cereal.messaging as messaging
 | 
					
						
							|  |  |  | from common.conversions import Conversions as CV
 | 
					
						
							|  |  |  | from common.realtime import DT_CTRL
 | 
					
						
							|  |  |  | from selfdrive.locationd.calibrationd import MIN_SPEED_FILTER
 | 
					
						
							|  |  |  | from selfdrive.version import get_short_branch
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AlertSize = log.ControlsState.AlertSize
 | 
					
						
							|  |  |  | AlertStatus = log.ControlsState.AlertStatus
 | 
					
						
							|  |  |  | VisualAlert = car.CarControl.HUDControl.VisualAlert
 | 
					
						
							|  |  |  | AudibleAlert = car.CarControl.HUDControl.AudibleAlert
 | 
					
						
							|  |  |  | EventName = car.CarEvent.EventName
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Alert priorities
 | 
					
						
							|  |  |  | class Priority(IntEnum):
 | 
					
						
							|  |  |  |   LOWEST = 0
 | 
					
						
							|  |  |  |   LOWER = 1
 | 
					
						
							|  |  |  |   LOW = 2
 | 
					
						
							|  |  |  |   MID = 3
 | 
					
						
							|  |  |  |   HIGH = 4
 | 
					
						
							|  |  |  |   HIGHEST = 5
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Event types
 | 
					
						
							|  |  |  | class ET:
 | 
					
						
							|  |  |  |   ENABLE = 'enable'
 | 
					
						
							|  |  |  |   PRE_ENABLE = 'preEnable'
 | 
					
						
							|  |  |  |   OVERRIDE = 'override'
 | 
					
						
							|  |  |  |   NO_ENTRY = 'noEntry'
 | 
					
						
							|  |  |  |   WARNING = 'warning'
 | 
					
						
							|  |  |  |   USER_DISABLE = 'userDisable'
 | 
					
						
							|  |  |  |   SOFT_DISABLE = 'softDisable'
 | 
					
						
							|  |  |  |   IMMEDIATE_DISABLE = 'immediateDisable'
 | 
					
						
							|  |  |  |   PERMANENT = 'permanent'
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # get event name from enum
 | 
					
						
							|  |  |  | EVENT_NAME = {v: k for k, v in EventName.schema.enumerants.items()}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Events:
 | 
					
						
							|  |  |  |   def __init__(self):
 | 
					
						
							|  |  |  |     self.events: List[int] = []
 | 
					
						
							|  |  |  |     self.static_events: List[int] = []
 | 
					
						
							|  |  |  |     self.events_prev = dict.fromkeys(EVENTS.keys(), 0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @property
 | 
					
						
							|  |  |  |   def names(self) -> List[int]:
 | 
					
						
							|  |  |  |     return self.events
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def __len__(self) -> int:
 | 
					
						
							|  |  |  |     return len(self.events)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def add(self, event_name: int, static: bool=False) -> None:
 | 
					
						
							|  |  |  |     if static:
 | 
					
						
							|  |  |  |       self.static_events.append(event_name)
 | 
					
						
							|  |  |  |     self.events.append(event_name)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def clear(self) -> None:
 | 
					
						
							|  |  |  |     self.events_prev = {k: (v + 1 if k in self.events else 0) for k, v in self.events_prev.items()}
 | 
					
						
							|  |  |  |     self.events = self.static_events.copy()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def any(self, event_type: str) -> bool:
 | 
					
						
							|  |  |  |     return any(event_type in EVENTS.get(e, {}) for e in self.events)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def create_alerts(self, event_types: List[str], callback_args=None):
 | 
					
						
							|  |  |  |     if callback_args is None:
 | 
					
						
							|  |  |  |       callback_args = []
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = []
 | 
					
						
							|  |  |  |     for e in self.events:
 | 
					
						
							|  |  |  |       types = EVENTS[e].keys()
 | 
					
						
							|  |  |  |       for et in event_types:
 | 
					
						
							|  |  |  |         if et in types:
 | 
					
						
							|  |  |  |           alert = EVENTS[e][et]
 | 
					
						
							|  |  |  |           if not isinstance(alert, Alert):
 | 
					
						
							|  |  |  |             alert = alert(*callback_args)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if DT_CTRL * (self.events_prev[e] + 1) >= alert.creation_delay:
 | 
					
						
							|  |  |  |             alert.alert_type = f"{EVENT_NAME[e]}/{et}"
 | 
					
						
							|  |  |  |             alert.event_type = et
 | 
					
						
							|  |  |  |             ret.append(alert)
 | 
					
						
							|  |  |  |     return ret
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def add_from_msg(self, events):
 | 
					
						
							|  |  |  |     for e in events:
 | 
					
						
							|  |  |  |       self.events.append(e.name.raw)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def to_msg(self):
 | 
					
						
							|  |  |  |     ret = []
 | 
					
						
							|  |  |  |     for event_name in self.events:
 | 
					
						
							|  |  |  |       event = car.CarEvent.new_message()
 | 
					
						
							|  |  |  |       event.name = event_name
 | 
					
						
							|  |  |  |       for event_type in EVENTS.get(event_name, {}):
 | 
					
						
							|  |  |  |         setattr(event, event_type, True)
 | 
					
						
							|  |  |  |       ret.append(event)
 | 
					
						
							|  |  |  |     return ret
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Alert:
 | 
					
						
							|  |  |  |   def __init__(self,
 | 
					
						
							|  |  |  |                alert_text_1: str,
 | 
					
						
							|  |  |  |                alert_text_2: str,
 | 
					
						
							|  |  |  |                alert_status: log.ControlsState.AlertStatus,
 | 
					
						
							|  |  |  |                alert_size: log.ControlsState.AlertSize,
 | 
					
						
							|  |  |  |                priority: Priority,
 | 
					
						
							|  |  |  |                visual_alert: car.CarControl.HUDControl.VisualAlert,
 | 
					
						
							|  |  |  |                audible_alert: car.CarControl.HUDControl.AudibleAlert,
 | 
					
						
							|  |  |  |                duration: float,
 | 
					
						
							|  |  |  |                alert_rate: float = 0.,
 | 
					
						
							|  |  |  |                creation_delay: float = 0.):
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     self.alert_text_1 = alert_text_1
 | 
					
						
							|  |  |  |     self.alert_text_2 = alert_text_2
 | 
					
						
							|  |  |  |     self.alert_status = alert_status
 | 
					
						
							|  |  |  |     self.alert_size = alert_size
 | 
					
						
							|  |  |  |     self.priority = priority
 | 
					
						
							|  |  |  |     self.visual_alert = visual_alert
 | 
					
						
							|  |  |  |     self.audible_alert = audible_alert
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     self.duration = int(duration / DT_CTRL)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     self.alert_rate = alert_rate
 | 
					
						
							|  |  |  |     self.creation_delay = creation_delay
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     self.alert_type = ""
 | 
					
						
							|  |  |  |     self.event_type: Optional[str] = None
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def __str__(self) -> str:
 | 
					
						
							|  |  |  |     return f"{self.alert_text_1}/{self.alert_text_2} {self.priority} {self.visual_alert} {self.audible_alert}"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def __gt__(self, alert2) -> bool:
 | 
					
						
							|  |  |  |     if not isinstance(alert2, Alert):
 | 
					
						
							|  |  |  |       return False
 | 
					
						
							|  |  |  |     return self.priority > alert2.priority
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NoEntryAlert(Alert):
 | 
					
						
							|  |  |  |   def __init__(self, alert_text_2: str,
 | 
					
						
							|  |  |  |                alert_text_1: str = "openpilot Unavailable",
 | 
					
						
							|  |  |  |                visual_alert: car.CarControl.HUDControl.VisualAlert=VisualAlert.none):
 | 
					
						
							|  |  |  |     super().__init__(alert_text_1, alert_text_2, AlertStatus.normal,
 | 
					
						
							|  |  |  |                      AlertSize.mid, Priority.LOW, visual_alert,
 | 
					
						
							|  |  |  |                      AudibleAlert.refuse, 3.)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class SoftDisableAlert(Alert):
 | 
					
						
							|  |  |  |   def __init__(self, alert_text_2: str):
 | 
					
						
							|  |  |  |     super().__init__("TAKE CONTROL IMMEDIATELY", alert_text_2,
 | 
					
						
							|  |  |  |                      AlertStatus.userPrompt, AlertSize.full,
 | 
					
						
							|  |  |  |                      Priority.MID, VisualAlert.steerRequired,
 | 
					
						
							|  |  |  |                      AudibleAlert.warningSoft, 2.),
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # less harsh version of SoftDisable, where the condition is user-triggered
 | 
					
						
							|  |  |  | class UserSoftDisableAlert(SoftDisableAlert):
 | 
					
						
							|  |  |  |   def __init__(self, alert_text_2: str):
 | 
					
						
							|  |  |  |     super().__init__(alert_text_2),
 | 
					
						
							|  |  |  |     self.alert_text_1 = "openpilot will disengage"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ImmediateDisableAlert(Alert):
 | 
					
						
							|  |  |  |   def __init__(self, alert_text_2: str):
 | 
					
						
							|  |  |  |     super().__init__("TAKE CONTROL IMMEDIATELY", alert_text_2,
 | 
					
						
							|  |  |  |                      AlertStatus.critical, AlertSize.full,
 | 
					
						
							|  |  |  |                      Priority.HIGHEST, VisualAlert.steerRequired,
 | 
					
						
							|  |  |  |                      AudibleAlert.warningImmediate, 4.),
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class EngagementAlert(Alert):
 | 
					
						
							|  |  |  |   def __init__(self, audible_alert: car.CarControl.HUDControl.AudibleAlert):
 | 
					
						
							|  |  |  |     super().__init__("", "",
 | 
					
						
							|  |  |  |                      AlertStatus.normal, AlertSize.none,
 | 
					
						
							|  |  |  |                      Priority.MID, VisualAlert.none,
 | 
					
						
							|  |  |  |                      audible_alert, .2),
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NormalPermanentAlert(Alert):
 | 
					
						
							|  |  |  |   def __init__(self, alert_text_1: str, alert_text_2: str = "", duration: float = 0.2, priority: Priority = Priority.LOWER, creation_delay: float = 0.):
 | 
					
						
							|  |  |  |     super().__init__(alert_text_1, alert_text_2,
 | 
					
						
							|  |  |  |                      AlertStatus.normal, AlertSize.mid if len(alert_text_2) else AlertSize.small,
 | 
					
						
							|  |  |  |                      priority, VisualAlert.none, AudibleAlert.none, duration, creation_delay=creation_delay),
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class StartupAlert(Alert):
 | 
					
						
							|  |  |  |   def __init__(self, alert_text_1: str, alert_text_2: str = "Always keep hands on wheel and eyes on road", alert_status=AlertStatus.normal):
 | 
					
						
							|  |  |  |     super().__init__(alert_text_1, alert_text_2,
 | 
					
						
							|  |  |  |                      alert_status, AlertSize.mid,
 | 
					
						
							|  |  |  |                      Priority.LOWER, VisualAlert.none, AudibleAlert.none, 10.),
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ********** helper functions **********
 | 
					
						
							|  |  |  | def get_display_speed(speed_ms: float, metric: bool) -> str:
 | 
					
						
							|  |  |  |   speed = int(round(speed_ms * (CV.MS_TO_KPH if metric else CV.MS_TO_MPH)))
 | 
					
						
							|  |  |  |   unit = 'km/h' if metric else 'mph'
 | 
					
						
							|  |  |  |   return f"{speed} {unit}"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ********** alert callback functions **********
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AlertCallbackType = Callable[[car.CarParams, car.CarState, messaging.SubMaster, bool, int], Alert]
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def soft_disable_alert(alert_text_2: str) -> AlertCallbackType:
 | 
					
						
							|  |  |  |   def func(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |     if soft_disable_time < int(0.5 / DT_CTRL):
 | 
					
						
							|  |  |  |       return ImmediateDisableAlert(alert_text_2)
 | 
					
						
							|  |  |  |     return SoftDisableAlert(alert_text_2)
 | 
					
						
							|  |  |  |   return func
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def user_soft_disable_alert(alert_text_2: str) -> AlertCallbackType:
 | 
					
						
							|  |  |  |   def func(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |     if soft_disable_time < int(0.5 / DT_CTRL):
 | 
					
						
							|  |  |  |       return ImmediateDisableAlert(alert_text_2)
 | 
					
						
							|  |  |  |     return UserSoftDisableAlert(alert_text_2)
 | 
					
						
							|  |  |  |   return func
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def startup_master_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   branch = get_short_branch("")
 | 
					
						
							|  |  |  |   if "REPLAY" in os.environ:
 | 
					
						
							|  |  |  |     branch = "replay"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return StartupAlert("WARNING: This branch is not tested", branch, alert_status=AlertStatus.userPrompt)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def below_engage_speed_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   return NoEntryAlert(f"Speed Below {get_display_speed(CP.minEnableSpeed, metric)}")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def below_steer_speed_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   return Alert(
 | 
					
						
							|  |  |  |     f"Steer Unavailable Below {get_display_speed(CP.minSteerSpeed, metric)}",
 | 
					
						
							|  |  |  |     "",
 | 
					
						
							|  |  |  |     AlertStatus.userPrompt, AlertSize.small,
 | 
					
						
							|  |  |  |     Priority.MID, VisualAlert.steerRequired, AudibleAlert.prompt, 0.4)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def calibration_incomplete_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   return Alert(
 | 
					
						
							|  |  |  |     "Calibration in Progress: %d%%" % sm['liveCalibration'].calPerc,
 | 
					
						
							|  |  |  |     f"Drive Above {get_display_speed(MIN_SPEED_FILTER, metric)}",
 | 
					
						
							|  |  |  |     AlertStatus.normal, AlertSize.mid,
 | 
					
						
							|  |  |  |     Priority.LOWEST, VisualAlert.none, AudibleAlert.none, .2)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def no_gps_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   gps_integrated = sm['peripheralState'].pandaType in (log.PandaState.PandaType.uno, log.PandaState.PandaType.dos)
 | 
					
						
							|  |  |  |   return Alert(
 | 
					
						
							|  |  |  |     "Poor GPS reception",
 | 
					
						
							|  |  |  |     "Hardware malfunctioning if sky is visible" if gps_integrated else "Check GPS antenna placement",
 | 
					
						
							|  |  |  |     AlertStatus.normal, AlertSize.mid,
 | 
					
						
							|  |  |  |     Priority.LOWER, VisualAlert.none, AudibleAlert.none, .2, creation_delay=300.)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # *** debug alerts ***
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def out_of_space_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   full_perc = round(100. - sm['deviceState'].freeSpacePercent)
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("Out of Storage", f"{full_perc}% full")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def posenet_invalid_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   mdl = sm['modelV2'].velocity.x[0] if len(sm['modelV2'].velocity.x) else math.nan
 | 
					
						
							|  |  |  |   err = CS.vEgo - mdl
 | 
					
						
							|  |  |  |   msg = f"Speed Error: {err:.1f} m/s"
 | 
					
						
							|  |  |  |   return NoEntryAlert(msg, alert_text_1="Posenet Speed Invalid")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def process_not_running_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   not_running = [p.name for p in sm['managerState'].processes if not p.running and p.shouldBeRunning]
 | 
					
						
							|  |  |  |   msg = ', '.join(not_running)
 | 
					
						
							|  |  |  |   return NoEntryAlert(msg, alert_text_1="Process Not Running")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def comm_issue_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   bs = [s for s in sm.data.keys() if not sm.all_checks([s, ])]
 | 
					
						
							|  |  |  |   msg = ', '.join(bs[:4])  # can't fit too many on one line
 | 
					
						
							|  |  |  |   return NoEntryAlert(msg, alert_text_1="Communication Issue Between Processes")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def camera_malfunction_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   all_cams = ('roadCameraState', 'driverCameraState', 'wideRoadCameraState')
 | 
					
						
							|  |  |  |   bad_cams = [s for s in all_cams if s in sm.data.keys() and not sm.all_checks([s, ])]
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("Camera Malfunction", ', '.join(bad_cams))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def calibration_invalid_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   rpy = sm['liveCalibration'].rpyCalib
 | 
					
						
							|  |  |  |   yaw = math.degrees(rpy[2] if len(rpy) == 3 else math.nan)
 | 
					
						
							|  |  |  |   pitch = math.degrees(rpy[1] if len(rpy) == 3 else math.nan)
 | 
					
						
							|  |  |  |   angles = f"Pitch: {pitch:.1f}°, Yaw: {yaw:.1f}°"
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("Calibration Invalid", angles)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def overheat_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   cpu = max(sm['deviceState'].cpuTempC, default=0.)
 | 
					
						
							|  |  |  |   gpu = max(sm['deviceState'].gpuTempC, default=0.)
 | 
					
						
							|  |  |  |   temp = max((cpu, gpu, sm['deviceState'].memoryTempC))
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("System Overheated", f"{temp:.0f} °C")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def low_memory_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("Low Memory", f"{sm['deviceState'].memoryUsagePercent}% used")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def high_cpu_usage_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   x = max(sm['deviceState'].cpuUsagePercent, default=0.)
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("High CPU Usage", f"{x}% used")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def modeld_lagging_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("Driving Model Lagging", f"{sm['modelV2'].frameDropPerc:.1f}% frames dropped")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def wrong_car_mode_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   text = "Cruise Mode Disabled"
 | 
					
						
							|  |  |  |   if CP.carName == "honda":
 | 
					
						
							|  |  |  |     text = "Main Switch Off"
 | 
					
						
							|  |  |  |   return NoEntryAlert(text)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def joystick_alert(CP: car.CarParams, CS: car.CarState, sm: messaging.SubMaster, metric: bool, soft_disable_time: int) -> Alert:
 | 
					
						
							|  |  |  |   axes = sm['testJoystick'].axes
 | 
					
						
							|  |  |  |   gb, steer = list(axes)[:2] if len(axes) else (0., 0.)
 | 
					
						
							|  |  |  |   vals = f"Gas: {round(gb * 100.)}%, Steer: {round(steer * 100.)}%"
 | 
					
						
							|  |  |  |   return NormalPermanentAlert("Joystick Mode", vals)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | EVENTS: Dict[int, Dict[str, Union[Alert, AlertCallbackType]]] = {
 | 
					
						
							|  |  |  |   # ********** events with no alerts **********
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.stockFcw: {},
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # ********** events only containing alerts displayed in all states **********
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.joystickDebug: {
 | 
					
						
							|  |  |  |     ET.WARNING: joystick_alert,
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Joystick Mode"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.controlsInitializing: {
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("System Initializing"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.startup: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: StartupAlert("Be ready to take over at any time")
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.startupMaster: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: startup_master_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Car is recognized, but marked as dashcam only
 | 
					
						
							|  |  |  |   EventName.startupNoControl: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: StartupAlert("Dashcam mode"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Car is not recognized
 | 
					
						
							|  |  |  |   EventName.startupNoCar: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: StartupAlert("Dashcam mode for unsupported car"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.startupNoFw: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: StartupAlert("Car Unrecognized",
 | 
					
						
							|  |  |  |                                "Check comma power connections",
 | 
					
						
							|  |  |  |                                alert_status=AlertStatus.userPrompt),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.dashcamMode: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Dashcam Mode",
 | 
					
						
							|  |  |  |                                        priority=Priority.LOWEST),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.invalidLkasSetting: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Stock LKAS is on",
 | 
					
						
							|  |  |  |                                        "Turn off stock LKAS to engage"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.cruiseMismatch: {
 | 
					
						
							|  |  |  |     #ET.PERMANENT: ImmediateDisableAlert("openpilot failed to cancel cruise"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # openpilot doesn't recognize the car. This switches openpilot into a
 | 
					
						
							|  |  |  |   # read-only mode. This can be solved by adding your fingerprint.
 | 
					
						
							|  |  |  |   # See https://github.com/commaai/openpilot/wiki/Fingerprinting for more information
 | 
					
						
							|  |  |  |   EventName.carUnrecognized: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Dashcam Mode",
 | 
					
						
							|  |  |  |                                        "Car Unrecognized",
 | 
					
						
							|  |  |  |                                        priority=Priority.LOWEST),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.stockAeb: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "BRAKE!",
 | 
					
						
							|  |  |  |       "Stock AEB: Risk of Collision",
 | 
					
						
							|  |  |  |       AlertStatus.critical, AlertSize.full,
 | 
					
						
							|  |  |  |       Priority.HIGHEST, VisualAlert.fcw, AudibleAlert.none, 2.),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Stock AEB: Risk of Collision"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.fcw: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "BRAKE!",
 | 
					
						
							|  |  |  |       "Risk of Collision",
 | 
					
						
							|  |  |  |       AlertStatus.critical, AlertSize.full,
 | 
					
						
							|  |  |  |       Priority.HIGHEST, VisualAlert.fcw, AudibleAlert.warningSoft, 2.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.ldw: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "Lane Departure Detected",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.ldw, AudibleAlert.prompt, 3.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # ********** events only containing alerts that display while engaged **********
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # openpilot tries to learn certain parameters about your car by observing
 | 
					
						
							|  |  |  |   # how the car behaves to steering inputs from both human and openpilot driving.
 | 
					
						
							|  |  |  |   # This includes:
 | 
					
						
							|  |  |  |   # - steer ratio: gear ratio of the steering rack. Steering angle divided by tire angle
 | 
					
						
							|  |  |  |   # - tire stiffness: how much grip your tires have
 | 
					
						
							|  |  |  |   # - angle offset: most steering angle sensors are offset and measure a non zero angle when driving straight
 | 
					
						
							|  |  |  |   # This alert is thrown when any of these values exceed a sanity check. This can be caused by
 | 
					
						
							|  |  |  |   # bad alignment or bad sensor data. If this happens consistently consider creating an issue on GitHub
 | 
					
						
							|  |  |  |   EventName.vehicleModelInvalid: {
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Vehicle Parameter Identification Failed"),
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Vehicle Parameter Identification Failed"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.steerTempUnavailableSilent: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Steering Temporarily Unavailable",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.steerRequired, AudibleAlert.prompt, 1.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.preDriverDistracted: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Pay Attention",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.promptDriverDistracted: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Pay Attention",
 | 
					
						
							|  |  |  |       "Driver Distracted",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.MID, VisualAlert.steerRequired, AudibleAlert.promptDistracted, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.driverDistracted: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "DISENGAGE IMMEDIATELY",
 | 
					
						
							|  |  |  |       "Driver Distracted",
 | 
					
						
							|  |  |  |       AlertStatus.critical, AlertSize.full,
 | 
					
						
							|  |  |  |       Priority.HIGH, VisualAlert.steerRequired, AudibleAlert.warningImmediate, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.preDriverUnresponsive: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Touch Steering Wheel: No Face Detected",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.steerRequired, AudibleAlert.none, .1, alert_rate=0.75),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.promptDriverUnresponsive: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Touch Steering Wheel",
 | 
					
						
							|  |  |  |       "Driver Unresponsive",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.MID, VisualAlert.steerRequired, AudibleAlert.promptDistracted, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.driverUnresponsive: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "DISENGAGE IMMEDIATELY",
 | 
					
						
							|  |  |  |       "Driver Unresponsive",
 | 
					
						
							|  |  |  |       AlertStatus.critical, AlertSize.full,
 | 
					
						
							|  |  |  |       Priority.HIGH, VisualAlert.steerRequired, AudibleAlert.warningImmediate, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.manualRestart: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "TAKE CONTROL",
 | 
					
						
							|  |  |  |       "Resume Driving Manually",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, .2),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.resumeRequired: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "STOPPED",
 | 
					
						
							|  |  |  |       "Press Resume to Go",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, .2),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.belowSteerSpeed: {
 | 
					
						
							|  |  |  |     ET.WARNING: below_steer_speed_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.preLaneChangeLeft: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Steer Left to Start Lane Change Once Safe",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, .1, alert_rate=0.75),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.preLaneChangeRight: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Steer Right to Start Lane Change Once Safe",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, .1, alert_rate=0.75),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.laneChangeBlocked: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Car Detected in Blindspot",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.prompt, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.laneChange: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Changing Lanes",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.steerSaturated: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Take Control",
 | 
					
						
							|  |  |  |       "Turn Exceeds Steering Limit",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.steerRequired, AudibleAlert.promptRepeat, 1.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Thrown when the fan is driven at >50% but is not rotating
 | 
					
						
							|  |  |  |   EventName.fanMalfunction: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Fan Malfunction", "Likely Hardware Issue"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Camera is not outputting frames
 | 
					
						
							|  |  |  |   EventName.cameraMalfunction: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: camera_malfunction_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  |   # Camera framerate too low
 | 
					
						
							|  |  |  |   EventName.cameraFrameRate: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Camera Frame Rate Low", "Reboot your Device"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Unused
 | 
					
						
							|  |  |  |   EventName.gpsMalfunction: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("GPS Malfunction", "Likely Hardware Issue"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # When the GPS position and localizer diverge the localizer is reset to the
 | 
					
						
							|  |  |  |   # current GPS position. This alert is thrown when the localizer is reset
 | 
					
						
							|  |  |  |   # more often than expected.
 | 
					
						
							|  |  |  |   EventName.localizerMalfunction: {
 | 
					
						
							|  |  |  |     # ET.PERMANENT: NormalPermanentAlert("Sensor Malfunction", "Hardware Malfunction"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # ********** events that affect controls state transitions **********
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.pcmEnable: {
 | 
					
						
							|  |  |  |     ET.ENABLE: EngagementAlert(AudibleAlert.engage),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.buttonEnable: {
 | 
					
						
							|  |  |  |     ET.ENABLE: EngagementAlert(AudibleAlert.engage),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.pcmDisable: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.buttonCancel: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.brakeHold: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Brake Hold Active"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.parkBrake: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Parking Brake Engaged"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.pedalPressed: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Pedal Pressed",
 | 
					
						
							|  |  |  |                               visual_alert=VisualAlert.brakePressed),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.pedalPressedPreEnable: {
 | 
					
						
							|  |  |  |     ET.PRE_ENABLE: Alert(
 | 
					
						
							|  |  |  |       "Release Pedal to Engage",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOWEST, VisualAlert.none, AudibleAlert.none, .1, creation_delay=1.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.gasPressedOverride: {
 | 
					
						
							|  |  |  |     ET.OVERRIDE: Alert(
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.none,
 | 
					
						
							|  |  |  |       Priority.LOWEST, VisualAlert.none, AudibleAlert.none, .1),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.wrongCarMode: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: wrong_car_mode_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.wrongCruiseMode: {
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: EngagementAlert(AudibleAlert.disengage),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Adaptive Cruise Disabled"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.steerTempUnavailable: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Steering Temporarily Unavailable"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Steering Temporarily Unavailable"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.outOfSpace: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: out_of_space_alert,
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Out of Storage"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.belowEngageSpeed: {
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: below_engage_speed_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.sensorDataInvalid: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "No Data from Device Sensors",
 | 
					
						
							|  |  |  |       "Reboot your Device",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.LOWER, VisualAlert.none, AudibleAlert.none, .2, creation_delay=1.),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("No Data from Device Sensors"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.noGps: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: no_gps_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.soundsUnavailable: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Speaker not found", "Reboot your Device"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Speaker not found"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.tooDistracted: {
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Distraction Level Too High"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.overheat: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: overheat_alert,
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("System Overheated"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("System Overheated"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.wrongGear: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: user_soft_disable_alert("Gear not D"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Gear not D"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # This alert is thrown when the calibration angles are outside of the acceptable range.
 | 
					
						
							|  |  |  |   # For example if the device is pointed too much to the left or the right.
 | 
					
						
							|  |  |  |   # Usually this can only be solved by removing the mount from the windshield completely,
 | 
					
						
							|  |  |  |   # and attaching while making sure the device is pointed straight forward and is level.
 | 
					
						
							|  |  |  |   # See https://comma.ai/setup for more information
 | 
					
						
							|  |  |  |   EventName.calibrationInvalid: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: calibration_invalid_alert,
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Calibration Invalid: Remount Device & Recalibrate"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Calibration Invalid: Remount Device & Recalibrate"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.calibrationIncomplete: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: calibration_incomplete_alert,
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Calibration in Progress"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Calibration in Progress"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.doorOpen: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: user_soft_disable_alert("Door Open"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Door Open"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.seatbeltNotLatched: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: user_soft_disable_alert("Seatbelt Unlatched"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Seatbelt Unlatched"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.espDisabled: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("ESP Off"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("ESP Off"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.lowBattery: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Low Battery"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Low Battery"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Different openpilot services communicate between each other at a certain
 | 
					
						
							|  |  |  |   # interval. If communication does not follow the regular schedule this alert
 | 
					
						
							|  |  |  |   # is thrown. This can mean a service crashed, did not broadcast a message for
 | 
					
						
							|  |  |  |   # ten times the regular interval, or the average interval is more than 10% too high.
 | 
					
						
							|  |  |  |   EventName.commIssue: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Communication Issue between Processes"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: comm_issue_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  |   EventName.commIssueAvgFreq: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Low Communication Rate between Processes"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Low Communication Rate between Processes"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.controlsdLagging: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Controls Lagging"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Controls Process Lagging: Reboot Your Device"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Thrown when manager detects a service exited unexpectedly while driving
 | 
					
						
							|  |  |  |   EventName.processNotRunning: {
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: process_not_running_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.radarFault: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Radar Error: Restart the Car"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Radar Error: Restart the Car"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Every frame from the camera should be processed by the model. If modeld
 | 
					
						
							|  |  |  |   # is not processing frames fast enough they have to be dropped. This alert is
 | 
					
						
							|  |  |  |   # thrown when over 20% of frames are dropped.
 | 
					
						
							|  |  |  |   EventName.modeldLagging: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Driving Model Lagging"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Driving Model Lagging"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: modeld_lagging_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Besides predicting the path, lane lines and lead car data the model also
 | 
					
						
							|  |  |  |   # predicts the current velocity and rotation speed of the car. If the model is
 | 
					
						
							|  |  |  |   # very uncertain about the current velocity while the car is moving, this
 | 
					
						
							|  |  |  |   # usually means the model has trouble understanding the scene. This is used
 | 
					
						
							|  |  |  |   # as a heuristic to warn the driver.
 | 
					
						
							|  |  |  |   EventName.posenetInvalid: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Posenet Speed Invalid"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: posenet_invalid_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # When the localizer detects an acceleration of more than 40 m/s^2 (~4G) we
 | 
					
						
							|  |  |  |   # alert the driver the device might have fallen from the windshield.
 | 
					
						
							|  |  |  |   EventName.deviceFalling: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Device Fell Off Mount"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Device Fell Off Mount"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.lowMemory: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("Low Memory: Reboot Your Device"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: low_memory_alert,
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Low Memory: Reboot Your Device"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.highCpuUsage: {
 | 
					
						
							|  |  |  |     #ET.SOFT_DISABLE: soft_disable_alert("System Malfunction: Reboot Your Device"),
 | 
					
						
							|  |  |  |     #ET.PERMANENT: NormalPermanentAlert("System Malfunction", "Reboot your Device"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: high_cpu_usage_alert,
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.accFaulted: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("Cruise Faulted"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Cruise Faulted", ""),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Cruise Faulted"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.controlsMismatch: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("Controls Mismatch"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Controls Mismatch"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.roadCameraError: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Camera CRC Error - Road",
 | 
					
						
							|  |  |  |                                        duration=1.,
 | 
					
						
							|  |  |  |                                        creation_delay=30.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.wideRoadCameraError: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Camera CRC Error - Road Fisheye",
 | 
					
						
							|  |  |  |                                        duration=1.,
 | 
					
						
							|  |  |  |                                        creation_delay=30.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.driverCameraError: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Camera CRC Error - Driver",
 | 
					
						
							|  |  |  |                                        duration=1.,
 | 
					
						
							|  |  |  |                                        creation_delay=30.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Sometimes the USB stack on the device can get into a bad state
 | 
					
						
							|  |  |  |   # causing the connection to the panda to be lost
 | 
					
						
							|  |  |  |   EventName.usbError: {
 | 
					
						
							|  |  |  |     ET.SOFT_DISABLE: soft_disable_alert("USB Error: Reboot Your Device"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("USB Error: Reboot Your Device", ""),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("USB Error: Reboot Your Device"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # This alert can be thrown for the following reasons:
 | 
					
						
							|  |  |  |   # - No CAN data received at all
 | 
					
						
							|  |  |  |   # - CAN data is received, but some message are not received at the right frequency
 | 
					
						
							|  |  |  |   # If you're not writing a new car port, this is usually cause by faulty wiring
 | 
					
						
							|  |  |  |   EventName.canError: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("CAN Error"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "CAN Error: Check Connections",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, 1., creation_delay=1.),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("CAN Error: Check Connections"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.canBusMissing: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("CAN Bus Disconnected"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "CAN Bus Disconnected: Likely Faulty Cable",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.small,
 | 
					
						
							|  |  |  |       Priority.LOW, VisualAlert.none, AudibleAlert.none, 1., creation_delay=1.),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("CAN Bus Disconnected: Check Connections"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.steerUnavailable: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("LKAS Fault: Restart the Car"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("LKAS Fault: Restart the car to engage"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("LKAS Fault: Restart the Car"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.brakeUnavailable: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("Cruise Fault: Restart the Car"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Cruise Fault: Restart the car to engage"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Cruise Fault: Restart the Car"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.reverseGear: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: Alert(
 | 
					
						
							|  |  |  |       "Reverse\nGear",
 | 
					
						
							|  |  |  |       "",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.full,
 | 
					
						
							|  |  |  |       Priority.LOWEST, VisualAlert.none, AudibleAlert.none, .2, creation_delay=0.5),
 | 
					
						
							|  |  |  |     ET.USER_DISABLE: ImmediateDisableAlert("Reverse Gear"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Reverse Gear"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # On cars that use stock ACC the car can decide to cancel ACC for various reasons.
 | 
					
						
							|  |  |  |   # When this happens we can no long control the car so the user needs to be warned immediately.
 | 
					
						
							|  |  |  |   EventName.cruiseDisabled: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("Cruise Is Off"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # For planning the trajectory Model Predictive Control (MPC) is used. This is
 | 
					
						
							|  |  |  |   # an optimization algorithm that is not guaranteed to find a feasible solution.
 | 
					
						
							|  |  |  |   # If no solution is found or the solution has a very high cost this alert is thrown.
 | 
					
						
							|  |  |  |   EventName.plannerError: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("Planner Solution Error"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Planner Solution Error"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # When the relay in the harness box opens the CAN bus between the LKAS camera
 | 
					
						
							|  |  |  |   # and the rest of the car is separated. When messages from the LKAS camera
 | 
					
						
							|  |  |  |   # are received on the car side this usually means the relay hasn't opened correctly
 | 
					
						
							|  |  |  |   # and this alert is thrown.
 | 
					
						
							|  |  |  |   EventName.relayMalfunction: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: ImmediateDisableAlert("Harness Relay Malfunction"),
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Harness Relay Malfunction", "Check Hardware"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Harness Relay Malfunction"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.noTarget: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: Alert(
 | 
					
						
							|  |  |  |       "openpilot Canceled",
 | 
					
						
							|  |  |  |       "No close lead car",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.HIGH, VisualAlert.none, AudibleAlert.disengage, 3.),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("No Close Lead Car"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.speedTooLow: {
 | 
					
						
							|  |  |  |     ET.IMMEDIATE_DISABLE: Alert(
 | 
					
						
							|  |  |  |       "openpilot Canceled",
 | 
					
						
							|  |  |  |       "Speed too low",
 | 
					
						
							|  |  |  |       AlertStatus.normal, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.HIGH, VisualAlert.none, AudibleAlert.disengage, 3.),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # When the car is driving faster than most cars in the training data, the model outputs can be unpredictable.
 | 
					
						
							|  |  |  |   EventName.speedTooHigh: {
 | 
					
						
							|  |  |  |     ET.WARNING: Alert(
 | 
					
						
							|  |  |  |       "Speed Too High",
 | 
					
						
							|  |  |  |       "Model uncertain at this speed",
 | 
					
						
							|  |  |  |       AlertStatus.userPrompt, AlertSize.mid,
 | 
					
						
							|  |  |  |       Priority.HIGH, VisualAlert.steerRequired, AudibleAlert.promptRepeat, 4.),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Slow down to engage"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.lowSpeedLockout: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("Cruise Fault: Restart the car to engage"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("Cruise Fault: Restart the Car"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   EventName.lkasDisabled: {
 | 
					
						
							|  |  |  |     ET.PERMANENT: NormalPermanentAlert("LKAS Disabled: Enable LKAS to engage"),
 | 
					
						
							|  |  |  |     ET.NO_ENTRY: NoEntryAlert("LKAS Disabled"),
 | 
					
						
							|  |  |  |   },
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }
 |