test_config_flow.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724
  1. """Tests for the config flow."""
  2. import pytest
  3. import voluptuous as vol
  4. from homeassistant.const import CONF_HOST, CONF_NAME
  5. from homeassistant.data_entry_flow import FlowResultType
  6. from pytest_homeassistant_custom_component.common import MockConfigEntry
  7. from custom_components.tuya_local import (
  8. async_migrate_entry,
  9. config_flow,
  10. get_device_unique_id,
  11. )
  12. from custom_components.tuya_local.const import (
  13. CONF_DEVICE_CID,
  14. CONF_DEVICE_ID,
  15. CONF_LOCAL_KEY,
  16. CONF_POLL_ONLY,
  17. CONF_PROTOCOL_VERSION,
  18. CONF_TYPE,
  19. DOMAIN,
  20. )
  21. # Designed to contain "special" characters that users constantly suspect.
  22. TESTKEY = ")<jO<@)'P1|kR$Kd"
  23. @pytest.fixture(autouse=True)
  24. def auto_enable_custom_integrations(enable_custom_integrations):
  25. yield
  26. @pytest.fixture(autouse=True)
  27. def prevent_task_creation(mocker):
  28. mocker.patch("custom_components.tuya_local.device.TuyaLocalDevice.register_entity")
  29. yield
  30. @pytest.fixture
  31. def bypass_setup(mocker):
  32. """Prevent actual setup of the integration after config flow."""
  33. mocker.patch("custom_components.tuya_local.async_setup_entry", return_value=True)
  34. yield
  35. @pytest.fixture
  36. def bypass_data_fetch(mocker):
  37. """Prevent actual data fetching from the device."""
  38. mocker.patch("tinytuya.Device.status", return_value={"1": True})
  39. yield
  40. @pytest.mark.asyncio
  41. async def test_init_entry(hass, bypass_data_fetch):
  42. """Test initialisation of the config flow."""
  43. entry = MockConfigEntry(
  44. domain=DOMAIN,
  45. version=11,
  46. title="test",
  47. data={
  48. CONF_DEVICE_ID: "deviceid",
  49. CONF_HOST: "hostname",
  50. CONF_LOCAL_KEY: TESTKEY,
  51. CONF_POLL_ONLY: False,
  52. CONF_PROTOCOL_VERSION: "auto",
  53. CONF_TYPE: "kogan_kahtp_heater",
  54. CONF_DEVICE_CID: None,
  55. },
  56. options={},
  57. )
  58. entry.add_to_hass(hass)
  59. await hass.config_entries.async_setup(entry.entry_id)
  60. await hass.async_block_till_done()
  61. assert hass.states.get("climate.test")
  62. assert hass.states.get("lock.test_child_lock")
  63. @pytest.mark.asyncio
  64. async def test_migrate_entry(hass, mocker):
  65. """Test migration from old entry format."""
  66. mock_device = mocker.MagicMock()
  67. mock_device.async_inferred_type = mocker.AsyncMock(
  68. return_value="goldair_gpph_heater"
  69. )
  70. mocker.patch("custom_components.tuya_local.setup_device", return_value=mock_device)
  71. entry = MockConfigEntry(
  72. domain=DOMAIN,
  73. version=1,
  74. title="test",
  75. data={
  76. CONF_DEVICE_ID: "deviceid",
  77. CONF_HOST: "hostname",
  78. CONF_LOCAL_KEY: TESTKEY,
  79. CONF_TYPE: "auto",
  80. "climate": True,
  81. "child_lock": True,
  82. "display_light": True,
  83. },
  84. )
  85. entry.add_to_hass(hass)
  86. assert await async_migrate_entry(hass, entry)
  87. mock_device.async_inferred_type = mocker.AsyncMock(return_value=None)
  88. mock_device.reset_mock()
  89. entry = MockConfigEntry(
  90. domain=DOMAIN,
  91. version=1,
  92. title="test2",
  93. data={
  94. CONF_DEVICE_ID: "deviceid",
  95. CONF_HOST: "hostname",
  96. CONF_LOCAL_KEY: TESTKEY,
  97. CONF_TYPE: "unknown",
  98. "climate": False,
  99. },
  100. )
  101. entry.add_to_hass(hass)
  102. assert not await async_migrate_entry(hass, entry)
  103. mock_device.reset_mock()
  104. entry = MockConfigEntry(
  105. domain=DOMAIN,
  106. version=2,
  107. title="test3",
  108. data={
  109. CONF_DEVICE_ID: "deviceid",
  110. CONF_HOST: "hostname",
  111. CONF_LOCAL_KEY: TESTKEY,
  112. CONF_TYPE: "auto",
  113. },
  114. options={
  115. "climate": False,
  116. },
  117. )
  118. entry.add_to_hass(hass)
  119. assert not await async_migrate_entry(hass, entry)
  120. mock_device.async_inferred_type = mocker.AsyncMock(return_value="smartplugv1")
  121. mock_device.reset_mock()
  122. entry = MockConfigEntry(
  123. domain=DOMAIN,
  124. version=3,
  125. title="test4",
  126. data={
  127. CONF_DEVICE_ID: "deviceid",
  128. CONF_HOST: "hostname",
  129. CONF_LOCAL_KEY: TESTKEY,
  130. CONF_TYPE: "smartplugv1",
  131. },
  132. options={
  133. "switch": True,
  134. },
  135. )
  136. entry.add_to_hass(hass)
  137. assert await async_migrate_entry(hass, entry)
  138. mock_device.async_inferred_type = mocker.AsyncMock(return_value="smartplugv2")
  139. mock_device.reset_mock()
  140. entry = MockConfigEntry(
  141. domain=DOMAIN,
  142. version=3,
  143. title="test5",
  144. data={
  145. CONF_DEVICE_ID: "deviceid",
  146. CONF_HOST: "hostname",
  147. CONF_LOCAL_KEY: TESTKEY,
  148. CONF_TYPE: "smartplugv1",
  149. },
  150. options={
  151. "switch": True,
  152. },
  153. )
  154. entry.add_to_hass(hass)
  155. assert await async_migrate_entry(hass, entry)
  156. mock_device.async_inferred_type = mocker.AsyncMock(
  157. return_value="goldair_dehumidifier"
  158. )
  159. mock_device.reset_mock()
  160. entry = MockConfigEntry(
  161. domain=DOMAIN,
  162. version=4,
  163. title="test6",
  164. data={
  165. CONF_DEVICE_ID: "deviceid",
  166. CONF_HOST: "hostname",
  167. CONF_LOCAL_KEY: TESTKEY,
  168. CONF_TYPE: "goldair_dehumidifier",
  169. },
  170. options={
  171. "humidifier": True,
  172. "fan": True,
  173. "light": True,
  174. "lock": False,
  175. "switch": True,
  176. },
  177. )
  178. entry.add_to_hass(hass)
  179. assert await async_migrate_entry(hass, entry)
  180. mock_device.async_inferred_type = mocker.AsyncMock(
  181. return_value="grid_connect_usb_double_power_point"
  182. )
  183. mock_device.reset_mock()
  184. entry = MockConfigEntry(
  185. domain=DOMAIN,
  186. version=6,
  187. title="test7",
  188. data={
  189. CONF_DEVICE_ID: "deviceid",
  190. CONF_HOST: "hostname",
  191. CONF_LOCAL_KEY: TESTKEY,
  192. CONF_TYPE: "grid_connect_usb_double_power_point",
  193. },
  194. options={
  195. "switch_main_switch": True,
  196. "switch_left_outlet": True,
  197. "switch_right_outlet": True,
  198. },
  199. )
  200. entry.add_to_hass(hass)
  201. assert await async_migrate_entry(hass, entry)
  202. @pytest.mark.asyncio
  203. async def test_flow_user_init(hass, mocker):
  204. """Test the initialisation of the form in the first page of the manual config flow path."""
  205. result = await hass.config_entries.flow.async_init(
  206. DOMAIN, context={"source": "local"}
  207. )
  208. expected = {
  209. "data_schema": mocker.ANY,
  210. "description_placeholders": mocker.ANY,
  211. "errors": {},
  212. "flow_id": mocker.ANY,
  213. "handler": DOMAIN,
  214. "step_id": "local",
  215. "type": "form",
  216. "last_step": mocker.ANY,
  217. "preview": mocker.ANY,
  218. }
  219. assert expected == result
  220. # Check the schema. Simple comparison does not work since they are not
  221. # the same object
  222. try:
  223. result["data_schema"](
  224. {CONF_DEVICE_ID: "test", CONF_LOCAL_KEY: TESTKEY, CONF_HOST: "test"}
  225. )
  226. except vol.MultipleInvalid:
  227. assert False
  228. try:
  229. result["data_schema"]({CONF_DEVICE_ID: "missing_some"})
  230. assert False
  231. except vol.MultipleInvalid:
  232. pass
  233. @pytest.mark.asyncio
  234. async def test_async_test_connection_valid(hass, mocker):
  235. """Test that device is returned when connection is valid."""
  236. mock_device = mocker.patch(
  237. "custom_components.tuya_local.config_flow.TuyaLocalDevice"
  238. )
  239. mock_instance = mocker.AsyncMock()
  240. mock_instance.has_returned_state = True
  241. mock_instance.pause = mocker.MagicMock()
  242. mock_instance.resume = mocker.MagicMock()
  243. mock_device.return_value = mock_instance
  244. hass.data[DOMAIN] = {"deviceid": {"device": mock_instance}}
  245. device = await config_flow.async_test_connection(
  246. {
  247. CONF_DEVICE_ID: "deviceid",
  248. CONF_LOCAL_KEY: TESTKEY,
  249. CONF_HOST: "hostname",
  250. CONF_PROTOCOL_VERSION: "auto",
  251. },
  252. hass,
  253. )
  254. assert device == mock_instance
  255. mock_instance.pause.assert_called_once()
  256. mock_instance.resume.assert_called_once()
  257. @pytest.mark.asyncio
  258. async def test_async_test_connection_for_subdevice_valid(hass, mocker):
  259. """Test that subdevice is returned when connection is valid."""
  260. mock_device = mocker.patch(
  261. "custom_components.tuya_local.config_flow.TuyaLocalDevice"
  262. )
  263. mock_instance = mocker.AsyncMock()
  264. mock_instance.has_returned_state = True
  265. mock_instance.pause = mocker.MagicMock()
  266. mock_instance.resume = mocker.MagicMock()
  267. mock_device.return_value = mock_instance
  268. hass.data[DOMAIN] = {"subdeviceid": {"device": mock_instance}}
  269. device = await config_flow.async_test_connection(
  270. {
  271. CONF_DEVICE_ID: "deviceid",
  272. CONF_LOCAL_KEY: TESTKEY,
  273. CONF_HOST: "hostname",
  274. CONF_PROTOCOL_VERSION: "auto",
  275. CONF_DEVICE_CID: "subdeviceid",
  276. },
  277. hass,
  278. )
  279. assert device == mock_instance
  280. mock_instance.pause.assert_called_once()
  281. mock_instance.resume.assert_called_once()
  282. @pytest.mark.asyncio
  283. async def test_async_test_connection_invalid(hass, mocker):
  284. """Test that None is returned when connection is invalid."""
  285. mock_device = mocker.patch(
  286. "custom_components.tuya_local.config_flow.TuyaLocalDevice"
  287. )
  288. mock_instance = mocker.AsyncMock()
  289. mock_instance.has_returned_state = False
  290. mock_device.return_value = mock_instance
  291. device = await config_flow.async_test_connection(
  292. {
  293. CONF_DEVICE_ID: "deviceid",
  294. CONF_LOCAL_KEY: TESTKEY,
  295. CONF_HOST: "hostname",
  296. CONF_PROTOCOL_VERSION: "auto",
  297. },
  298. hass,
  299. )
  300. assert device is None
  301. @pytest.mark.asyncio
  302. async def test_flow_user_init_invalid_config(hass, mocker):
  303. """Test errors populated when config is invalid."""
  304. mocker.patch(
  305. "custom_components.tuya_local.config_flow.async_test_connection",
  306. return_value=None,
  307. )
  308. flow = await hass.config_entries.flow.async_init(
  309. DOMAIN, context={"source": "local"}
  310. )
  311. result = await hass.config_entries.flow.async_configure(
  312. flow["flow_id"],
  313. user_input={
  314. CONF_DEVICE_ID: "deviceid",
  315. CONF_HOST: "hostname",
  316. CONF_LOCAL_KEY: "badkey",
  317. CONF_PROTOCOL_VERSION: "auto",
  318. CONF_POLL_ONLY: False,
  319. },
  320. )
  321. assert {"base": "connection"} == result["errors"]
  322. def setup_device_mock(mock, mocker, failure=False, type="test"):
  323. mock_type = mocker.MagicMock()
  324. mock_type.legacy_type = type
  325. mock_type.config_type = type
  326. mock_type.match_quality.return_value = 100
  327. mock.async_possible_types = mocker.AsyncMock(
  328. return_value=[mock_type] if not failure else []
  329. )
  330. @pytest.mark.asyncio
  331. async def test_flow_user_init_data_valid(hass, mocker):
  332. """Test we advance to the next step when connection config is valid."""
  333. mock_device = mocker.MagicMock()
  334. setup_device_mock(mock_device, mocker)
  335. mocker.patch(
  336. "custom_components.tuya_local.config_flow.async_test_connection",
  337. return_value=mock_device,
  338. )
  339. flow = await hass.config_entries.flow.async_init(
  340. DOMAIN, context={"source": "local"}
  341. )
  342. result = await hass.config_entries.flow.async_configure(
  343. flow["flow_id"],
  344. user_input={
  345. CONF_DEVICE_ID: "deviceid",
  346. CONF_HOST: "hostname",
  347. CONF_LOCAL_KEY: TESTKEY,
  348. },
  349. )
  350. assert "form" == result["type"]
  351. assert "select_type" == result["step_id"]
  352. @pytest.mark.asyncio
  353. async def test_flow_select_type_init(hass, mocker):
  354. """Test the initialisation of the form in the 2nd step of the config flow."""
  355. mock_device = mocker.patch.object(config_flow.ConfigFlowHandler, "device")
  356. setup_device_mock(mock_device, mocker)
  357. result = await hass.config_entries.flow.async_init(
  358. DOMAIN, context={"source": "select_type"}
  359. )
  360. expected = {
  361. "data_schema": mocker.ANY,
  362. "description_placeholders": None,
  363. "errors": None,
  364. "flow_id": mocker.ANY,
  365. "handler": DOMAIN,
  366. "step_id": "select_type",
  367. "type": "form",
  368. "last_step": mocker.ANY,
  369. "preview": mocker.ANY,
  370. }
  371. assert expected == result
  372. # Check the schema. Simple comparison does not work since they are not
  373. # the same object
  374. try:
  375. result["data_schema"]({CONF_TYPE: "test"})
  376. except vol.MultipleInvalid:
  377. assert False
  378. try:
  379. result["data_schema"]({CONF_TYPE: "not_test"})
  380. assert False
  381. except vol.MultipleInvalid:
  382. pass
  383. @pytest.mark.asyncio
  384. async def test_flow_select_type_aborts_when_no_match(hass, mocker):
  385. """Test the flow aborts when an unsupported device is used."""
  386. mock_device = mocker.patch.object(config_flow.ConfigFlowHandler, "device")
  387. setup_device_mock(mock_device, mocker, failure=True)
  388. result = await hass.config_entries.flow.async_init(
  389. DOMAIN, context={"source": "select_type"}
  390. )
  391. assert result["type"] == "abort"
  392. assert result["reason"] == "not_supported"
  393. @pytest.mark.asyncio
  394. async def test_flow_select_type_data_valid(hass, mocker):
  395. """Test the flow continues when valid data is supplied."""
  396. mock_device = mocker.patch.object(config_flow.ConfigFlowHandler, "device")
  397. setup_device_mock(mock_device, mocker, type="smartplugv1")
  398. flow = await hass.config_entries.flow.async_init(
  399. DOMAIN, context={"source": "select_type"}
  400. )
  401. result = await hass.config_entries.flow.async_configure(
  402. flow["flow_id"],
  403. user_input={CONF_TYPE: "smartplugv1"},
  404. )
  405. assert "form" == result["type"]
  406. assert "choose_entities" == result["step_id"]
  407. @pytest.mark.asyncio
  408. async def test_flow_choose_entities_init(hass, mocker):
  409. """Test the initialisation of the form in the 3rd step of the config flow."""
  410. mocker.patch.dict(config_flow.ConfigFlowHandler.data, {CONF_TYPE: "smartplugv1"})
  411. result = await hass.config_entries.flow.async_init(
  412. DOMAIN, context={"source": "choose_entities"}
  413. )
  414. expected = {
  415. "data_schema": mocker.ANY,
  416. "description_placeholders": None,
  417. "errors": None,
  418. "flow_id": mocker.ANY,
  419. "handler": DOMAIN,
  420. "step_id": "choose_entities",
  421. "type": "form",
  422. "last_step": mocker.ANY,
  423. "preview": mocker.ANY,
  424. }
  425. assert expected == result
  426. # Check the schema. Simple comparison does not work since they are not
  427. # the same object
  428. try:
  429. result["data_schema"]({CONF_NAME: "test"})
  430. except vol.MultipleInvalid:
  431. assert False
  432. try:
  433. result["data_schema"]({"climate": True})
  434. assert False
  435. except vol.MultipleInvalid:
  436. pass
  437. @pytest.mark.asyncio
  438. async def test_flow_choose_entities_creates_config_entry(hass, bypass_setup, mocker):
  439. """Test the flow ends when data is valid."""
  440. mocker.patch.dict(
  441. config_flow.ConfigFlowHandler.data,
  442. {
  443. CONF_DEVICE_ID: "deviceid",
  444. CONF_LOCAL_KEY: TESTKEY,
  445. CONF_HOST: "hostname",
  446. CONF_POLL_ONLY: False,
  447. CONF_PROTOCOL_VERSION: "auto",
  448. CONF_TYPE: "kogan_kahtp_heater",
  449. CONF_DEVICE_CID: None,
  450. },
  451. )
  452. flow = await hass.config_entries.flow.async_init(
  453. DOMAIN, context={"source": "choose_entities"}
  454. )
  455. result = await hass.config_entries.flow.async_configure(
  456. flow["flow_id"],
  457. user_input={
  458. CONF_NAME: "test",
  459. },
  460. )
  461. expected = {
  462. "version": 13,
  463. "minor_version": mocker.ANY,
  464. "context": {"source": "choose_entities"},
  465. "type": FlowResultType.CREATE_ENTRY,
  466. "flow_id": mocker.ANY,
  467. "handler": DOMAIN,
  468. "title": "test",
  469. "description": None,
  470. "description_placeholders": None,
  471. "result": mocker.ANY,
  472. "subentries": (),
  473. "options": {},
  474. "data": {
  475. CONF_DEVICE_ID: "deviceid",
  476. CONF_HOST: "hostname",
  477. CONF_LOCAL_KEY: TESTKEY,
  478. CONF_POLL_ONLY: False,
  479. CONF_PROTOCOL_VERSION: "auto",
  480. CONF_TYPE: "kogan_kahtp_heater",
  481. CONF_DEVICE_CID: None,
  482. },
  483. }
  484. assert expected == result
  485. @pytest.mark.asyncio
  486. async def test_options_flow_init(hass, bypass_data_fetch):
  487. """Test config flow options."""
  488. config_entry = MockConfigEntry(
  489. domain=DOMAIN,
  490. version=13,
  491. unique_id="uniqueid",
  492. data={
  493. CONF_DEVICE_ID: "deviceid",
  494. CONF_HOST: "hostname",
  495. CONF_LOCAL_KEY: TESTKEY,
  496. CONF_NAME: "test",
  497. CONF_POLL_ONLY: False,
  498. CONF_PROTOCOL_VERSION: "auto",
  499. CONF_TYPE: "smartplugv1",
  500. CONF_DEVICE_CID: "",
  501. },
  502. )
  503. config_entry.add_to_hass(hass)
  504. assert await hass.config_entries.async_setup(config_entry.entry_id)
  505. await hass.async_block_till_done()
  506. # show initial form
  507. result = await hass.config_entries.options.async_init(config_entry.entry_id)
  508. assert "form" == result["type"]
  509. assert "user" == result["step_id"]
  510. assert {} == result["errors"]
  511. assert result["data_schema"](
  512. {
  513. CONF_HOST: "hostname",
  514. CONF_LOCAL_KEY: TESTKEY,
  515. }
  516. )
  517. @pytest.mark.asyncio
  518. async def test_options_flow_modifies_config(hass, bypass_setup, mocker):
  519. mock_device = mocker.MagicMock()
  520. mocker.patch(
  521. "custom_components.tuya_local.config_flow.async_test_connection",
  522. return_value=mock_device,
  523. )
  524. config_entry = MockConfigEntry(
  525. domain=DOMAIN,
  526. version=13,
  527. unique_id="uniqueid",
  528. data={
  529. CONF_DEVICE_ID: "deviceid",
  530. CONF_HOST: "hostname",
  531. CONF_LOCAL_KEY: TESTKEY,
  532. CONF_NAME: "test",
  533. CONF_POLL_ONLY: False,
  534. CONF_PROTOCOL_VERSION: "auto",
  535. CONF_TYPE: "ble_pt216_temp_humidity",
  536. CONF_DEVICE_CID: "subdeviceid",
  537. },
  538. )
  539. config_entry.add_to_hass(hass)
  540. assert await hass.config_entries.async_setup(config_entry.entry_id)
  541. await hass.async_block_till_done()
  542. # show initial form
  543. form = await hass.config_entries.options.async_init(config_entry.entry_id)
  544. # submit updated config
  545. result = await hass.config_entries.options.async_configure(
  546. form["flow_id"],
  547. user_input={
  548. CONF_HOST: "new_hostname",
  549. CONF_LOCAL_KEY: "new_key",
  550. CONF_POLL_ONLY: False,
  551. CONF_PROTOCOL_VERSION: 3.3,
  552. },
  553. )
  554. expected = {
  555. CONF_HOST: "new_hostname",
  556. CONF_LOCAL_KEY: "new_key",
  557. CONF_POLL_ONLY: False,
  558. CONF_PROTOCOL_VERSION: 3.3,
  559. }
  560. assert "create_entry" == result["type"]
  561. assert "" == result["title"]
  562. assert expected == result["data"]
  563. @pytest.mark.asyncio
  564. async def test_options_flow_fails_when_connection_fails(
  565. hass, bypass_data_fetch, mocker
  566. ):
  567. mocker.patch(
  568. "custom_components.tuya_local.config_flow.async_test_connection",
  569. return_value=None,
  570. )
  571. config_entry = MockConfigEntry(
  572. domain=DOMAIN,
  573. version=13,
  574. unique_id="uniqueid",
  575. data={
  576. CONF_DEVICE_ID: "deviceid",
  577. CONF_HOST: "hostname",
  578. CONF_LOCAL_KEY: TESTKEY,
  579. CONF_NAME: "test",
  580. CONF_POLL_ONLY: False,
  581. CONF_PROTOCOL_VERSION: "auto",
  582. CONF_TYPE: "smartplugv1",
  583. CONF_DEVICE_CID: "",
  584. },
  585. )
  586. config_entry.add_to_hass(hass)
  587. assert await hass.config_entries.async_setup(config_entry.entry_id)
  588. await hass.async_block_till_done()
  589. # show initial form
  590. form = await hass.config_entries.options.async_init(config_entry.entry_id)
  591. # submit updated config
  592. result = await hass.config_entries.options.async_configure(
  593. form["flow_id"],
  594. user_input={
  595. CONF_HOST: "new_hostname",
  596. CONF_LOCAL_KEY: "new_key",
  597. },
  598. )
  599. assert "form" == result["type"]
  600. assert "user" == result["step_id"]
  601. assert {"base": "connection"} == result["errors"]
  602. @pytest.mark.asyncio
  603. async def test_options_flow_fails_when_config_is_missing(hass, mocker):
  604. mock_device = mocker.MagicMock()
  605. mocker.patch(
  606. "custom_components.tuya_local.config_flow.async_test_connection",
  607. return_value=mock_device,
  608. )
  609. config_entry = MockConfigEntry(
  610. domain=DOMAIN,
  611. version=13,
  612. unique_id="uniqueid",
  613. data={
  614. CONF_DEVICE_ID: "deviceid",
  615. CONF_HOST: "hostname",
  616. CONF_LOCAL_KEY: TESTKEY,
  617. CONF_NAME: "test",
  618. CONF_POLL_ONLY: False,
  619. CONF_PROTOCOL_VERSION: "auto",
  620. CONF_TYPE: "non_existing",
  621. },
  622. )
  623. config_entry.add_to_hass(hass)
  624. await hass.config_entries.async_setup(config_entry.entry_id)
  625. await hass.async_block_till_done()
  626. # show initial form
  627. result = await hass.config_entries.options.async_init(config_entry.entry_id)
  628. assert result["type"] == "abort"
  629. assert result["reason"] == "not_supported"
  630. def test_migration_gets_correct_device_id():
  631. """Test that migration gets the correct device id."""
  632. # Normal device
  633. entry = MockConfigEntry(
  634. domain=DOMAIN,
  635. version=1,
  636. title="test",
  637. data={
  638. CONF_DEVICE_ID: "deviceid",
  639. CONF_HOST: "hostname",
  640. CONF_LOCAL_KEY: TESTKEY,
  641. CONF_TYPE: "auto",
  642. },
  643. )
  644. assert get_device_unique_id(entry) == "deviceid"