base_device_tests.py 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. from unittest import IsolatedAsyncioTestCase
  2. from unittest.mock import AsyncMock, Mock, PropertyMock, patch
  3. from uuid import uuid4
  4. from homeassistant.helpers.entity import EntityCategory
  5. from custom_components.tuya_local.alarm_control_panel import TuyaLocalAlarmControlPanel
  6. from custom_components.tuya_local.binary_sensor import TuyaLocalBinarySensor
  7. from custom_components.tuya_local.button import TuyaLocalButton
  8. from custom_components.tuya_local.camera import TuyaLocalCamera
  9. from custom_components.tuya_local.climate import TuyaLocalClimate
  10. from custom_components.tuya_local.cover import TuyaLocalCover
  11. from custom_components.tuya_local.datetime import TuyaLocalDateTime
  12. from custom_components.tuya_local.event import TuyaLocalEvent
  13. from custom_components.tuya_local.fan import TuyaLocalFan
  14. from custom_components.tuya_local.helpers.device_config import (
  15. TuyaDeviceConfig,
  16. possible_matches,
  17. )
  18. from custom_components.tuya_local.humidifier import TuyaLocalHumidifier
  19. from custom_components.tuya_local.infrared import TuyaLocalInfrared
  20. from custom_components.tuya_local.lawn_mower import TuyaLocalLawnMower
  21. from custom_components.tuya_local.light import TuyaLocalLight
  22. from custom_components.tuya_local.lock import TuyaLocalLock
  23. from custom_components.tuya_local.number import TuyaLocalNumber
  24. from custom_components.tuya_local.remote import TuyaLocalRemote
  25. from custom_components.tuya_local.select import TuyaLocalSelect
  26. from custom_components.tuya_local.sensor import TuyaLocalSensor
  27. from custom_components.tuya_local.siren import TuyaLocalSiren
  28. from custom_components.tuya_local.switch import TuyaLocalSwitch
  29. from custom_components.tuya_local.text import TuyaLocalText
  30. from custom_components.tuya_local.time import TuyaLocalTime
  31. from custom_components.tuya_local.vacuum import TuyaLocalVacuum
  32. from custom_components.tuya_local.valve import TuyaLocalValve
  33. from custom_components.tuya_local.water_heater import TuyaLocalWaterHeater
  34. DEVICE_TYPES = {
  35. "alarm_control_panel": TuyaLocalAlarmControlPanel,
  36. "binary_sensor": TuyaLocalBinarySensor,
  37. "button": TuyaLocalButton,
  38. "camera": TuyaLocalCamera,
  39. "climate": TuyaLocalClimate,
  40. "cover": TuyaLocalCover,
  41. "datetime": TuyaLocalDateTime,
  42. "event": TuyaLocalEvent,
  43. "fan": TuyaLocalFan,
  44. "humidifier": TuyaLocalHumidifier,
  45. "infrared": TuyaLocalInfrared,
  46. "lawn_mower": TuyaLocalLawnMower,
  47. "light": TuyaLocalLight,
  48. "lock": TuyaLocalLock,
  49. "number": TuyaLocalNumber,
  50. "remote": TuyaLocalRemote,
  51. "switch": TuyaLocalSwitch,
  52. "select": TuyaLocalSelect,
  53. "sensor": TuyaLocalSensor,
  54. "siren": TuyaLocalSiren,
  55. "text": TuyaLocalText,
  56. "time": TuyaLocalTime,
  57. "vacuum": TuyaLocalVacuum,
  58. "valve": TuyaLocalValve,
  59. "water_heater": TuyaLocalWaterHeater,
  60. }
  61. class TuyaDeviceTestCase(IsolatedAsyncioTestCase):
  62. __test__ = False
  63. def setUpForConfig(self, config_file, payload):
  64. """Perform setup tasks for every test."""
  65. device_patcher = patch("custom_components.tuya_local.device.TuyaLocalDevice")
  66. self.addCleanup(device_patcher.stop)
  67. self.mock_device = device_patcher.start()
  68. self.dps = payload.copy()
  69. self.mock_device.get_property.side_effect = lambda id: self.dps.get(id)
  70. cfg = TuyaDeviceConfig(config_file)
  71. self.conf_type = cfg.legacy_type
  72. type(self.mock_device).has_returned_state = PropertyMock(return_value=True)
  73. type(self.mock_device).unique_id = PropertyMock(return_value=str(uuid4()))
  74. self.mock_device.name = cfg.name
  75. self.entities = {}
  76. self.secondary_category = []
  77. self.names = {}
  78. for e in cfg.all_entities():
  79. self.entities[e.config_id] = self.create_entity(e)
  80. self.names[e.config_id] = e.name
  81. def create_entity(self, config):
  82. """Create an entity to match the config"""
  83. dev_type = DEVICE_TYPES[config.entity]
  84. if dev_type:
  85. entity = dev_type(self.mock_device, config)
  86. entity.platform = Mock()
  87. entity.platform.name = dev_type
  88. entity.platform.platform_translations = {}
  89. return entity
  90. def mark_secondary(self, entities):
  91. self.secondary_category = self.secondary_category + entities
  92. def test_config_matched(self):
  93. for cfg in possible_matches(self.dps):
  94. if cfg.legacy_type == self.conf_type:
  95. quality = cfg.match_quality(self.dps)
  96. self.assertEqual(
  97. quality,
  98. 100.0,
  99. msg=f"{self.conf_type} is an imperfect match at {quality}%",
  100. )
  101. return
  102. self.fail()
  103. def test_should_poll(self):
  104. for e in self.entities.values():
  105. self.assertFalse(e.should_poll)
  106. def test_available(self):
  107. for e in self.entities.values():
  108. self.assertTrue(e.available)
  109. def test_entity_category(self):
  110. for k, e in self.entities.items():
  111. if k in self.secondary_category:
  112. if type(e) in [TuyaLocalBinarySensor, TuyaLocalSensor]:
  113. self.assertEqual(
  114. e.entity_category,
  115. EntityCategory.DIAGNOSTIC,
  116. msg=f"{k} is {e.entity_category.value}, expected diagnostic",
  117. )
  118. elif type(e) is TuyaLocalButton:
  119. self.assertIn(
  120. e.entity_category,
  121. [EntityCategory.CONFIG, EntityCategory.DIAGNOSTIC],
  122. msg=f"{k} is unsupported {e.entity_category.value}",
  123. )
  124. else:
  125. self.assertEqual(
  126. e.entity_category,
  127. EntityCategory.CONFIG,
  128. msg=f"{k} is {e.entity_category.value}, expected config",
  129. )
  130. else:
  131. self.assertIsNone(
  132. e.entity_category,
  133. msg=f"{k} is {e.entity_category}, expected None",
  134. )
  135. # name has become more difficult to test with translation support, but it is working
  136. # in practice.
  137. # def test_name_returns_device_name(self):
  138. # for e in self.entities:
  139. # self.assertEqual(self.entities[e].name, self.names[e])
  140. def test_unique_id_contains_device_unique_id(self):
  141. entities = {}
  142. for e in self.entities.values():
  143. self.assertIn(self.mock_device.unique_id, e.unique_id)
  144. if type(e) not in entities:
  145. entities[type(e)] = []
  146. entities[type(e)].append(e.unique_id)
  147. for e in entities.values():
  148. self.assertCountEqual(e, set(e))
  149. def test_device_info_returns_device_info_from_device(self):
  150. for e in self.entities.values():
  151. self.assertEqual(e.device_info, self.mock_device.device_info)
  152. async def test_update(self):
  153. for e in self.entities.values():
  154. result = AsyncMock()
  155. self.mock_device.async_refresh.return_value = result()
  156. self.mock_device.async_refresh.reset_mock()
  157. await e.async_update()
  158. self.mock_device.async_refresh.assert_called_once()
  159. result.assert_awaited()