You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					426 lines
				
				16 KiB
			
		
		
			
		
	
	
					426 lines
				
				16 KiB
			| 
											2 years ago
										 | import re
 | ||
|  | from collections import namedtuple
 | ||
|  | import copy
 | ||
|  | from dataclasses import dataclass, field
 | ||
|  | from enum import Enum
 | ||
|  | 
 | ||
| 
											7 months ago
										 | from opendbc.car.common.conversions import Conversions as CV
 | ||
|  | from opendbc.car.structs import CarParams
 | ||
| 
											2 years ago
										 | 
 | ||
|  | GOOD_TORQUE_THRESHOLD = 1.0  # m/s^2
 | ||
|  | MODEL_YEARS_RE = r"(?<= )((\d{4}-\d{2})|(\d{4}))(,|$)"
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class Column(Enum):
 | ||
|  |   MAKE = "Make"
 | ||
|  |   MODEL = "Model"
 | ||
|  |   PACKAGE = "Supported Package"
 | ||
|  |   LONGITUDINAL = "ACC"
 | ||
|  |   FSR_LONGITUDINAL = "No ACC accel below"
 | ||
|  |   FSR_STEERING = "No ALC below"
 | ||
|  |   STEERING_TORQUE = "Steering Torque"
 | ||
|  |   AUTO_RESUME = "Resume from stop"
 | ||
|  |   HARDWARE = "Hardware Needed"
 | ||
|  |   VIDEO = "Video"
 | ||
|  | 
 | ||
|  | 
 | ||
| 
											7 months ago
										 | class ExtraCarsColumn(Enum):
 | ||
|  |   MAKE = "Make"
 | ||
|  |   MODEL = "Model"
 | ||
|  |   PACKAGE = "Package"
 | ||
|  |   SUPPORT = "Support Level"
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class SupportType(Enum):
 | ||
|  |   UPSTREAM = "Upstream"             # Actively maintained by comma, plug-and-play in release versions of openpilot
 | ||
|  |   REVIEW = "Under review"           # Dashcam, but planned for official support after safety validation
 | ||
|  |   DASHCAM = "Dashcam mode"          # Dashcam, but may be drivable in a community fork
 | ||
|  |   COMMUNITY = "Community"           # Not upstream, but available in a custom community fork, not validated by comma
 | ||
|  |   CUSTOM = "Custom"                 # Upstream, but don't have a harness available or need an unusual custom install
 | ||
|  |   INCOMPATIBLE = "Not compatible"   # Known fundamental incompatibility such as Flexray or hydraulic power steering
 | ||
|  | 
 | ||
|  | 
 | ||
| 
											2 years ago
										 | class Star(Enum):
 | ||
|  |   FULL = "full"
 | ||
|  |   HALF = "half"
 | ||
|  |   EMPTY = "empty"
 | ||
|  | 
 | ||
|  | 
 | ||
|  | # A part + its comprised parts
 | ||
|  | @dataclass
 | ||
|  | class BasePart:
 | ||
|  |   name: str
 | ||
| 
											1 year ago
										 |   parts: list[Enum] = field(default_factory=list)
 | ||
| 
											2 years ago
										 | 
 | ||
|  |   def all_parts(self):
 | ||
|  |     # Recursively get all parts
 | ||
|  |     _parts = 'parts'
 | ||
|  |     parts = []
 | ||
|  |     parts.extend(getattr(self, _parts))
 | ||
|  |     for part in getattr(self, _parts):
 | ||
|  |       parts.extend(part.value.all_parts())
 | ||
|  | 
 | ||
|  |     return parts
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class EnumBase(Enum):
 | ||
|  |   @property
 | ||
| 
											2 years ago
										 |   def part_type(self):
 | ||
| 
											2 years ago
										 |     return PartType(self.__class__)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class Mount(EnumBase):
 | ||
|  |   mount = BasePart("mount")
 | ||
|  |   angled_mount_8_degrees = BasePart("angled mount (8 degrees)")
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class Cable(EnumBase):
 | ||
|  |   long_obdc_cable = BasePart("long OBD-C cable")
 | ||
|  |   usb_a_2_a_cable = BasePart("USB A-A cable")
 | ||
|  |   usbc_otg_cable = BasePart("USB C OTG cable")
 | ||
|  |   usbc_coupler = BasePart("USB-C coupler")
 | ||
|  |   obd_c_cable_1_5ft = BasePart("OBD-C cable (1.5 ft)")
 | ||
|  |   right_angle_obd_c_cable_1_5ft = BasePart("right angle OBD-C cable (1.5 ft)")
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class Accessory(EnumBase):
 | ||
|  |   harness_box = BasePart("harness box")
 | ||
|  |   comma_power_v2 = BasePart("comma power v2")
 | ||
|  | 
 | ||
|  | 
 | ||
|  | @dataclass
 | ||
|  | class BaseCarHarness(BasePart):
 | ||
| 
											7 months ago
										 |   parts: list[Enum] = field(default_factory=lambda: [Accessory.harness_box, Accessory.comma_power_v2])
 | ||
| 
											2 years ago
										 |   has_connector: bool = True  # without are hidden on the harness connector page
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class CarHarness(EnumBase):
 | ||
|  |   nidec = BaseCarHarness("Honda Nidec connector")
 | ||
|  |   bosch_a = BaseCarHarness("Honda Bosch A connector")
 | ||
|  |   bosch_b = BaseCarHarness("Honda Bosch B connector")
 | ||
| 
											7 months ago
										 |   bosch_c = BaseCarHarness("Honda Bosch C connector")
 | ||
| 
											2 years ago
										 |   toyota_a = BaseCarHarness("Toyota A connector")
 | ||
|  |   toyota_b = BaseCarHarness("Toyota B connector")
 | ||
| 
											2 years ago
										 |   subaru_a = BaseCarHarness("Subaru A connector")
 | ||
|  |   subaru_b = BaseCarHarness("Subaru B connector")
 | ||
| 
											2 years ago
										 |   subaru_c = BaseCarHarness("Subaru C connector")
 | ||
|  |   subaru_d = BaseCarHarness("Subaru D connector")
 | ||
| 
											2 years ago
										 |   fca = BaseCarHarness("FCA connector")
 | ||
|  |   ram = BaseCarHarness("Ram connector")
 | ||
| 
											7 months ago
										 |   vw_a = BaseCarHarness("VW A connector")
 | ||
|  |   vw_j533 = BaseCarHarness("VW J533 connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler, Accessory.comma_power_v2])
 | ||
| 
											2 years ago
										 |   hyundai_a = BaseCarHarness("Hyundai A connector")
 | ||
|  |   hyundai_b = BaseCarHarness("Hyundai B connector")
 | ||
|  |   hyundai_c = BaseCarHarness("Hyundai C connector")
 | ||
|  |   hyundai_d = BaseCarHarness("Hyundai D connector")
 | ||
|  |   hyundai_e = BaseCarHarness("Hyundai E connector")
 | ||
|  |   hyundai_f = BaseCarHarness("Hyundai F connector")
 | ||
|  |   hyundai_g = BaseCarHarness("Hyundai G connector")
 | ||
|  |   hyundai_h = BaseCarHarness("Hyundai H connector")
 | ||
|  |   hyundai_i = BaseCarHarness("Hyundai I connector")
 | ||
|  |   hyundai_j = BaseCarHarness("Hyundai J connector")
 | ||
|  |   hyundai_k = BaseCarHarness("Hyundai K connector")
 | ||
|  |   hyundai_l = BaseCarHarness("Hyundai L connector")
 | ||
|  |   hyundai_m = BaseCarHarness("Hyundai M connector")
 | ||
|  |   hyundai_n = BaseCarHarness("Hyundai N connector")
 | ||
|  |   hyundai_o = BaseCarHarness("Hyundai O connector")
 | ||
|  |   hyundai_p = BaseCarHarness("Hyundai P connector")
 | ||
|  |   hyundai_q = BaseCarHarness("Hyundai Q connector")
 | ||
| 
											2 years ago
										 |   hyundai_r = BaseCarHarness("Hyundai R connector")
 | ||
| 
											2 years ago
										 |   custom = BaseCarHarness("Developer connector")
 | ||
| 
											7 months ago
										 |   obd_ii = BaseCarHarness("OBD-II connector", parts=[Cable.long_obdc_cable], has_connector=False)
 | ||
| 
											2 years ago
										 |   gm = BaseCarHarness("GM connector", parts=[Accessory.harness_box])
 | ||
| 
											7 months ago
										 |   gmsdgm = BaseCarHarness("GM SDGM connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler, Accessory.comma_power_v2])
 | ||
|  |   nissan_a = BaseCarHarness("Nissan A connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler])
 | ||
|  |   nissan_b = BaseCarHarness("Nissan B connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler])
 | ||
| 
											2 years ago
										 |   mazda = BaseCarHarness("Mazda connector")
 | ||
|  |   ford_q3 = BaseCarHarness("Ford Q3 connector")
 | ||
| 
											7 months ago
										 |   ford_q4 = BaseCarHarness("Ford Q4 connector", parts=[Accessory.harness_box, Accessory.comma_power_v2, Cable.long_obdc_cable, Cable.usbc_coupler])
 | ||
|  |   rivian = BaseCarHarness("Rivian A connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler])
 | ||
|  |   tesla_a = BaseCarHarness("Tesla A connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler])
 | ||
|  |   tesla_b = BaseCarHarness("Tesla B connector", parts=[Accessory.harness_box, Cable.long_obdc_cable, Cable.usbc_coupler])
 | ||
| 
											2 years ago
										 | 
 | ||
|  | 
 | ||
|  | class Device(EnumBase):
 | ||
| 
											2 years ago
										 |   threex = BasePart("comma 3X", parts=[Mount.mount, Cable.right_angle_obd_c_cable_1_5ft])
 | ||
|  |   # variant of comma 3X with angled mounts
 | ||
|  |   threex_angled_mount = BasePart("comma 3X", parts=[Mount.angled_mount_8_degrees, Cable.right_angle_obd_c_cable_1_5ft])
 | ||
| 
											2 years ago
										 |   red_panda = BasePart("red panda")
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class Kit(EnumBase):
 | ||
| 
											2 years ago
										 |   red_panda_kit = BasePart("CAN FD panda kit", parts=[Device.red_panda, Accessory.harness_box,
 | ||
|  |                                                       Cable.usb_a_2_a_cable, Cable.usbc_otg_cable, Cable.obd_c_cable_1_5ft])
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class Tool(EnumBase):
 | ||
|  |   socket_8mm_deep = BasePart("Socket Wrench 8mm or 5/16\" (deep)")
 | ||
|  |   pry_tool = BasePart("Pry Tool")
 | ||
| 
											2 years ago
										 | 
 | ||
|  | 
 | ||
|  | class PartType(Enum):
 | ||
|  |   accessory = Accessory
 | ||
|  |   cable = Cable
 | ||
|  |   connector = CarHarness
 | ||
|  |   device = Device
 | ||
|  |   kit = Kit
 | ||
|  |   mount = Mount
 | ||
| 
											2 years ago
										 |   tool = Tool
 | ||
| 
											2 years ago
										 | 
 | ||
|  | 
 | ||
| 
											1 year ago
										 | DEFAULT_CAR_PARTS: list[EnumBase] = [Device.threex]
 | ||
| 
											2 years ago
										 | 
 | ||
|  | 
 | ||
|  | @dataclass
 | ||
|  | class CarParts:
 | ||
| 
											1 year ago
										 |   parts: list[EnumBase] = field(default_factory=list)
 | ||
| 
											2 years ago
										 | 
 | ||
|  |   def __call__(self):
 | ||
|  |     return copy.deepcopy(self)
 | ||
|  | 
 | ||
|  |   @classmethod
 | ||
| 
											1 year ago
										 |   def common(cls, add: list[EnumBase] = None, remove: list[EnumBase] = None):
 | ||
| 
											2 years ago
										 |     p = [part for part in (add or []) + DEFAULT_CAR_PARTS if part not in (remove or [])]
 | ||
|  |     return cls(p)
 | ||
|  | 
 | ||
|  |   def all_parts(self):
 | ||
|  |     parts = []
 | ||
|  |     for part in self.parts:
 | ||
|  |       parts.extend(part.value.all_parts())
 | ||
|  |     return self.parts + parts
 | ||
|  | 
 | ||
|  | 
 | ||
|  | CarFootnote = namedtuple("CarFootnote", ["text", "column", "docs_only", "shop_footnote"], defaults=(False, False))
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class CommonFootnote(Enum):
 | ||
|  |   EXP_LONG_AVAIL = CarFootnote(
 | ||
| 
											2 years ago
										 |     "openpilot Longitudinal Control (Alpha) is available behind a toggle; " +
 | ||
| 
											7 months ago
										 |     "the toggle is only available in non-release branches such as `devel` or `nightly-dev`.",
 | ||
| 
											2 years ago
										 |     Column.LONGITUDINAL, docs_only=True)
 | ||
|  |   EXP_LONG_DSU = CarFootnote(
 | ||
| 
											2 years ago
										 |     "By default, this car will use the stock Adaptive Cruise Control (ACC) for longitudinal control. " +
 | ||
|  |     "If the Driver Support Unit (DSU) is disconnected, openpilot ACC will replace " +
 | ||
| 
											2 years ago
										 |     "stock ACC. <b><i>NOTE: disconnecting the DSU disables Automatic Emergency Braking (AEB).</i></b>",
 | ||
|  |     Column.LONGITUDINAL)
 | ||
|  | 
 | ||
|  | 
 | ||
| 
											1 year ago
										 | def get_footnotes(footnotes: list[Enum], column: Column) -> list[Enum]:
 | ||
| 
											2 years ago
										 |   # Returns applicable footnotes given current column
 | ||
|  |   return [fn for fn in footnotes if fn.value.column == column]
 | ||
|  | 
 | ||
|  | 
 | ||
|  | # TODO: store years as a list
 | ||
|  | def get_year_list(years):
 | ||
|  |   years_list = []
 | ||
|  |   if len(years) == 0:
 | ||
|  |     return years_list
 | ||
|  | 
 | ||
|  |   for year in years.split(','):
 | ||
|  |     year = year.strip()
 | ||
|  |     if len(year) == 4:
 | ||
|  |       years_list.append(str(year))
 | ||
|  |     elif "-" in year and len(year) == 7:
 | ||
|  |       start, end = year.split("-")
 | ||
|  |       years_list.extend(map(str, range(int(start), int(f"20{end}") + 1)))
 | ||
|  |     else:
 | ||
|  |       raise Exception(f"Malformed year string: {years}")
 | ||
|  |   return years_list
 | ||
|  | 
 | ||
|  | 
 | ||
| 
											1 year ago
										 | def split_name(name: str) -> tuple[str, str, str]:
 | ||
| 
											2 years ago
										 |   make, model = name.split(" ", 1)
 | ||
|  |   years = ""
 | ||
|  |   match = re.search(MODEL_YEARS_RE, model)
 | ||
|  |   if match is not None:
 | ||
|  |     years = model[match.start():]
 | ||
|  |     model = model[:match.start() - 1]
 | ||
|  |   return make, model, years
 | ||
|  | 
 | ||
|  | 
 | ||
|  | @dataclass
 | ||
| 
											1 year ago
										 | class CarDocs:
 | ||
| 
											2 years ago
										 |   # make + model + model years
 | ||
|  |   name: str
 | ||
|  | 
 | ||
|  |   # Example for Toyota Corolla MY20
 | ||
|  |   # requirements: Lane Tracing Assist (LTA) and Dynamic Radar Cruise Control (DRCC)
 | ||
|  |   # US Market reference: "All", since all Corolla in the US come standard with LTA and DRCC
 | ||
|  | 
 | ||
|  |   # the simplest description of the requirements for the US market
 | ||
|  |   package: str
 | ||
|  | 
 | ||
|  |   # the minimum compatibility requirements for this model, regardless
 | ||
|  |   # of market. can be a package, trim, or list of features
 | ||
| 
											1 year ago
										 |   requirements: str | None = None
 | ||
| 
											2 years ago
										 | 
 | ||
| 
											1 year ago
										 |   video_link: str | None = None
 | ||
|  |   footnotes: list[Enum] = field(default_factory=list)
 | ||
|  |   min_steer_speed: float | None = None
 | ||
|  |   min_enable_speed: float | None = None
 | ||
|  |   auto_resume: bool | None = None
 | ||
| 
											2 years ago
										 | 
 | ||
|  |   # all the parts needed for the supported car
 | ||
|  |   car_parts: CarParts = field(default_factory=CarParts)
 | ||
|  | 
 | ||
| 
											7 months ago
										 |   merged: bool = True
 | ||
|  |   support_type: SupportType = SupportType.UPSTREAM
 | ||
|  |   support_link: str | None = "#upstream"
 | ||
|  | 
 | ||
| 
											2 years ago
										 |   def __post_init__(self):
 | ||
|  |     self.make, self.model, self.years = split_name(self.name)
 | ||
|  |     self.year_list = get_year_list(self.years)
 | ||
|  | 
 | ||
| 
											7 months ago
										 |   def init(self, CP: CarParams, all_footnotes=None):
 | ||
|  |     self.brand = CP.brand
 | ||
| 
											2 years ago
										 |     self.car_fingerprint = CP.carFingerprint
 | ||
|  | 
 | ||
| 
											7 months ago
										 |     if self.merged and CP.dashcamOnly:
 | ||
|  |       if self.support_type != SupportType.REVIEW:
 | ||
|  |         self.support_type = SupportType.DASHCAM
 | ||
|  |         self.support_link = "#dashcam"
 | ||
|  |       else:
 | ||
|  |         self.support_link = "#under-review"
 | ||
|  | 
 | ||
| 
											2 years ago
										 |     # longitudinal column
 | ||
|  |     op_long = "Stock"
 | ||
|  |     if CP.experimentalLongitudinalAvailable or CP.enableDsu:
 | ||
|  |       op_long = "openpilot available"
 | ||
|  |       if CP.enableDsu:
 | ||
|  |         self.footnotes.append(CommonFootnote.EXP_LONG_DSU)
 | ||
|  |       else:
 | ||
|  |         self.footnotes.append(CommonFootnote.EXP_LONG_AVAIL)
 | ||
|  |     elif CP.openpilotLongitudinalControl and not CP.enableDsu:
 | ||
|  |       op_long = "openpilot"
 | ||
|  | 
 | ||
|  |     # min steer & enable speed columns
 | ||
|  |     # TODO: set all the min steer speeds in carParams and remove this
 | ||
|  |     if self.min_steer_speed is not None:
 | ||
| 
											1 year ago
										 |       assert CP.minSteerSpeed < 0.5, f"{CP.carFingerprint}: Minimum steer speed set in both CarDocs and CarParams"
 | ||
| 
											2 years ago
										 |     else:
 | ||
|  |       self.min_steer_speed = CP.minSteerSpeed
 | ||
|  | 
 | ||
|  |     # TODO: set all the min enable speeds in carParams correctly and remove this
 | ||
|  |     if self.min_enable_speed is None:
 | ||
|  |       self.min_enable_speed = CP.minEnableSpeed
 | ||
|  | 
 | ||
|  |     if self.auto_resume is None:
 | ||
| 
											1 year ago
										 |       self.auto_resume = CP.autoResumeSng and self.min_enable_speed <= 0
 | ||
| 
											2 years ago
										 | 
 | ||
|  |     # hardware column
 | ||
|  |     hardware_col = "None"
 | ||
|  |     if self.car_parts.parts:
 | ||
|  |       model_years = self.model + (' ' + self.years if self.years else '')
 | ||
| 
											2 years ago
										 |       buy_link = f'<a href="https://comma.ai/shop/comma-3x.html?make={self.make}&model={model_years}">Buy Here</a>'
 | ||
|  | 
 | ||
|  |       tools_docs = [part for part in self.car_parts.all_parts() if isinstance(part, Tool)]
 | ||
|  |       parts_docs = [part for part in self.car_parts.all_parts() if not isinstance(part, Tool)]
 | ||
|  | 
 | ||
|  |       def display_func(parts):
 | ||
|  |         return '<br>'.join([f"- {parts.count(part)} {part.value.name}" for part in sorted(set(parts), key=lambda part: str(part.value.name))])
 | ||
|  | 
 | ||
|  |       hardware_col = f'<details><summary>Parts</summary><sub>{display_func(parts_docs)}<br>{buy_link}</sub></details>'
 | ||
|  |       if len(tools_docs):
 | ||
|  |         hardware_col += f'<details><summary>Tools</summary><sub>{display_func(tools_docs)}</sub></details>'
 | ||
| 
											2 years ago
										 | 
 | ||
| 
											1 year ago
										 |     self.row: dict[Enum, str | Star] = {
 | ||
| 
											2 years ago
										 |       Column.MAKE: self.make,
 | ||
|  |       Column.MODEL: self.model,
 | ||
|  |       Column.PACKAGE: self.package,
 | ||
|  |       Column.LONGITUDINAL: op_long,
 | ||
|  |       Column.FSR_LONGITUDINAL: f"{max(self.min_enable_speed * CV.MS_TO_MPH, 0):.0f} mph",
 | ||
|  |       Column.FSR_STEERING: f"{max(self.min_steer_speed * CV.MS_TO_MPH, 0):.0f} mph",
 | ||
|  |       Column.STEERING_TORQUE: Star.EMPTY,
 | ||
|  |       Column.AUTO_RESUME: Star.FULL if self.auto_resume else Star.EMPTY,
 | ||
|  |       Column.HARDWARE: hardware_col,
 | ||
|  |       Column.VIDEO: self.video_link if self.video_link is not None else "",  # replaced with an image and link from template in get_column
 | ||
|  |     }
 | ||
|  | 
 | ||
| 
											7 months ago
										 |     if self.support_link is not None:
 | ||
|  |       support_info = f"[{self.support_type.value}]({self.support_link})"
 | ||
|  |     else:
 | ||
|  |       support_info = self.support_type.value
 | ||
|  | 
 | ||
|  |     self.extra_cars_row: dict[Enum, str] = {
 | ||
|  |       ExtraCarsColumn.MAKE: self.make,
 | ||
|  |       ExtraCarsColumn.MODEL: self.model,
 | ||
|  |       ExtraCarsColumn.PACKAGE: self.package,
 | ||
|  |       ExtraCarsColumn.SUPPORT: support_info,
 | ||
|  |     }
 | ||
|  | 
 | ||
| 
											2 years ago
										 |     # Set steering torque star from max lateral acceleration
 | ||
|  |     assert CP.maxLateralAccel > 0.1
 | ||
|  |     if CP.maxLateralAccel >= GOOD_TORQUE_THRESHOLD:
 | ||
|  |       self.row[Column.STEERING_TORQUE] = Star.FULL
 | ||
|  | 
 | ||
|  |     self.all_footnotes = all_footnotes
 | ||
|  |     self.detail_sentence = self.get_detail_sentence(CP)
 | ||
|  | 
 | ||
|  |     return self
 | ||
|  | 
 | ||
| 
											7 months ago
										 |   def init_make(self, CP: CarParams):
 | ||
| 
											1 year ago
										 |     """CarDocs subclasses can add make-specific logic for harness selection, footnotes, etc."""
 | ||
| 
											2 years ago
										 | 
 | ||
|  |   def get_detail_sentence(self, CP):
 | ||
|  |     if not CP.notCar:
 | ||
|  |       sentence_builder = "openpilot upgrades your <strong>{car_model}</strong> with automated lane centering{alc} and adaptive cruise control{acc}."
 | ||
|  | 
 | ||
|  |       if self.min_steer_speed > self.min_enable_speed:
 | ||
|  |         alc = f" <strong>above {self.min_steer_speed * CV.MS_TO_MPH:.0f} mph</strong>," if self.min_steer_speed > 0 else " <strong>at all speeds</strong>,"
 | ||
|  |       else:
 | ||
|  |         alc = ""
 | ||
|  | 
 | ||
|  |       # Exception for cars which do not auto-resume yet
 | ||
|  |       acc = ""
 | ||
|  |       if self.min_enable_speed > 0:
 | ||
|  |         acc = f" <strong>while driving above {self.min_enable_speed * CV.MS_TO_MPH:.0f} mph</strong>"
 | ||
|  |       elif self.auto_resume:
 | ||
|  |         acc = " <strong>that automatically resumes from a stop</strong>"
 | ||
|  | 
 | ||
|  |       if self.row[Column.STEERING_TORQUE] != Star.FULL:
 | ||
|  |         sentence_builder += " This car may not be able to take tight turns on its own."
 | ||
|  | 
 | ||
|  |       # experimental mode
 | ||
| 
											7 months ago
										 |       exp_link = "<a href='https://blog.comma.ai/090release/#experimental-mode' target='_blank' class='highlight'>Experimental mode</a>"
 | ||
| 
											1 year ago
										 |       if CP.openpilotLongitudinalControl and not CP.experimentalLongitudinalAvailable:
 | ||
| 
											2 years ago
										 |         sentence_builder += f" Traffic light and stop sign handling is also available in {exp_link}."
 | ||
|  | 
 | ||
|  |       return sentence_builder.format(car_model=f"{self.make} {self.model}", alc=alc, acc=acc)
 | ||
|  | 
 | ||
|  |     else:
 | ||
| 
											1 year ago
										 |       if CP.carFingerprint == "COMMA_BODY":
 | ||
| 
											7 months ago
										 |         return "The body is a robotics dev kit that can run openpilot. <a href='https://www.commabody.com' target='_blank' class='highlight'>Learn more.</a>"
 | ||
| 
											2 years ago
										 |       else:
 | ||
|  |         raise Exception(f"This notCar does not have a detail sentence: {CP.carFingerprint}")
 | ||
|  | 
 | ||
|  |   def get_column(self, column: Column, star_icon: str, video_icon: str, footnote_tag: str) -> str:
 | ||
| 
											1 year ago
										 |     item: str | Star = self.row[column]
 | ||
| 
											2 years ago
										 |     if isinstance(item, Star):
 | ||
|  |       item = star_icon.format(item.value)
 | ||
|  |     elif column == Column.MODEL and len(self.years):
 | ||
|  |       item += f" {self.years}"
 | ||
|  |     elif column == Column.VIDEO and len(item) > 0:
 | ||
|  |       item = video_icon.format(item)
 | ||
|  | 
 | ||
|  |     footnotes = get_footnotes(self.footnotes, column)
 | ||
|  |     if len(footnotes):
 | ||
|  |       sups = sorted([self.all_footnotes[fn] for fn in footnotes])
 | ||
|  |       item += footnote_tag.format(f'{",".join(map(str, sups))}')
 | ||
|  | 
 | ||
|  |     return item
 | ||
| 
											7 months ago
										 | 
 | ||
|  |   def get_extra_cars_column(self, column: ExtraCarsColumn) -> str:
 | ||
|  |     item: str = self.extra_cars_row[column]
 | ||
|  |     if column == ExtraCarsColumn.MODEL and len(self.years):
 | ||
|  |       item += f" {self.years}"
 | ||
|  | 
 | ||
|  |     return item
 | ||
|  | 
 | ||
|  | 
 | ||
|  | @dataclass
 | ||
|  | class ExtraCarDocs(CarDocs):
 | ||
|  |   package: str = "Any"
 | ||
|  |   merged: bool = False
 | ||
|  |   support_type: SupportType = SupportType.INCOMPATIBLE
 | ||
|  |   support_link: str | None = "#incompatible"
 |