2019-04-03 15:40:03 +00:00
|
|
|
"""Support for Frontier Silicon Devices (Medion, Hama, Auna,...)."""
|
2022-01-04 10:08:28 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2017-02-28 14:23:07 +00:00
|
|
|
import logging
|
2022-11-29 12:31:49 +00:00
|
|
|
from typing import Any
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-23 07:41:07 +00:00
|
|
|
from afsapi import (
|
|
|
|
AFSAPI,
|
|
|
|
ConnectionError as FSConnectionError,
|
|
|
|
NotImplementedException as FSNotImplementedException,
|
|
|
|
PlayState,
|
|
|
|
)
|
2017-02-28 14:23:07 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2022-04-06 08:55:25 +00:00
|
|
|
from homeassistant.components.media_player import (
|
|
|
|
PLATFORM_SCHEMA,
|
2022-11-29 12:31:49 +00:00
|
|
|
BrowseError,
|
|
|
|
BrowseMedia,
|
2022-04-06 08:55:25 +00:00
|
|
|
MediaPlayerEntity,
|
|
|
|
MediaPlayerEntityFeature,
|
2022-09-10 20:16:04 +00:00
|
|
|
MediaPlayerState,
|
|
|
|
MediaType,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2022-09-10 20:16:04 +00:00
|
|
|
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_PORT
|
2022-01-04 10:08:28 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2017-02-28 14:23:07 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2022-05-30 16:31:57 +00:00
|
|
|
from homeassistant.helpers.entity import DeviceInfo
|
2022-01-04 10:08:28 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-11-29 12:31:49 +00:00
|
|
|
from .browse_media import browse_node, browse_top_level
|
|
|
|
from .const import DEFAULT_PIN, DEFAULT_PORT, DOMAIN, MEDIA_CONTENT_ID_PRESET
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-05-30 16:31:57 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
2022-05-30 16:31:57 +00:00
|
|
|
vol.Optional(CONF_PASSWORD, default=DEFAULT_PIN): cv.string,
|
2020-02-23 16:05:24 +00:00
|
|
|
vol.Optional(CONF_NAME): cv.string,
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
|
|
|
)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
|
|
|
|
2022-01-04 10:08:28 +00:00
|
|
|
async def async_setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2017-03-26 13:50:40 +00:00
|
|
|
"""Set up the Frontier Silicon platform."""
|
2017-02-28 14:23:07 +00:00
|
|
|
if discovery_info is not None:
|
2022-05-30 16:31:57 +00:00
|
|
|
webfsapi_url = await AFSAPI.get_webfsapi_endpoint(
|
|
|
|
discovery_info["ssdp_description"]
|
|
|
|
)
|
|
|
|
afsapi = AFSAPI(webfsapi_url, DEFAULT_PIN)
|
|
|
|
|
|
|
|
name = await afsapi.get_friendly_name()
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities(
|
2022-05-30 16:31:57 +00:00
|
|
|
[AFSAPIDevice(name, afsapi)],
|
2020-02-23 16:05:24 +00:00
|
|
|
True,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2022-01-04 10:08:28 +00:00
|
|
|
return
|
2017-02-28 14:23:07 +00:00
|
|
|
|
|
|
|
host = config.get(CONF_HOST)
|
|
|
|
port = config.get(CONF_PORT)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
2020-02-23 16:05:24 +00:00
|
|
|
name = config.get(CONF_NAME)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
|
|
|
try:
|
2022-05-30 16:31:57 +00:00
|
|
|
webfsapi_url = await AFSAPI.get_webfsapi_endpoint(
|
|
|
|
f"http://{host}:{port}/device"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2022-05-30 16:31:57 +00:00
|
|
|
except FSConnectionError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Could not add the FSAPI device at %s:%s -> %s", host, port, password
|
|
|
|
)
|
2022-05-31 07:56:25 +00:00
|
|
|
return
|
|
|
|
afsapi = AFSAPI(webfsapi_url, password)
|
|
|
|
async_add_entities([AFSAPIDevice(name, afsapi)], True)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
class AFSAPIDevice(MediaPlayerEntity):
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Representation of a Frontier Silicon device on the network."""
|
|
|
|
|
2022-11-29 12:31:49 +00:00
|
|
|
_attr_media_content_type: str = MediaType.CHANNEL
|
2022-05-31 09:56:44 +00:00
|
|
|
|
2022-04-06 08:55:25 +00:00
|
|
|
_attr_supported_features = (
|
|
|
|
MediaPlayerEntityFeature.PAUSE
|
|
|
|
| MediaPlayerEntityFeature.VOLUME_SET
|
|
|
|
| MediaPlayerEntityFeature.VOLUME_MUTE
|
|
|
|
| MediaPlayerEntityFeature.VOLUME_STEP
|
|
|
|
| MediaPlayerEntityFeature.PREVIOUS_TRACK
|
|
|
|
| MediaPlayerEntityFeature.NEXT_TRACK
|
|
|
|
| MediaPlayerEntityFeature.SEEK
|
|
|
|
| MediaPlayerEntityFeature.PLAY_MEDIA
|
|
|
|
| MediaPlayerEntityFeature.PLAY
|
|
|
|
| MediaPlayerEntityFeature.STOP
|
|
|
|
| MediaPlayerEntityFeature.TURN_ON
|
|
|
|
| MediaPlayerEntityFeature.TURN_OFF
|
|
|
|
| MediaPlayerEntityFeature.SELECT_SOURCE
|
2022-06-28 21:13:43 +00:00
|
|
|
| MediaPlayerEntityFeature.SELECT_SOUND_MODE
|
2022-11-29 12:31:49 +00:00
|
|
|
| MediaPlayerEntityFeature.BROWSE_MEDIA
|
2022-04-06 08:55:25 +00:00
|
|
|
)
|
|
|
|
|
2022-05-30 16:31:57 +00:00
|
|
|
def __init__(self, name: str | None, afsapi: AFSAPI) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Initialize the Frontier Silicon API device."""
|
2022-05-30 16:31:57 +00:00
|
|
|
self.fs_device = afsapi
|
|
|
|
|
|
|
|
self._attr_device_info = DeviceInfo(
|
|
|
|
identifiers={(DOMAIN, afsapi.webfsapi_endpoint)},
|
|
|
|
name=name,
|
|
|
|
)
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_name = name
|
2022-05-30 16:31:57 +00:00
|
|
|
|
2022-08-23 07:41:07 +00:00
|
|
|
self._max_volume: int | None = None
|
|
|
|
|
|
|
|
self.__modes_by_label: dict[str, str] | None = None
|
|
|
|
self.__sound_modes_by_label: dict[str, str] | None = None
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-23 07:41:07 +00:00
|
|
|
self._supports_sound_mode: bool = True
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-26 08:54:22 +00:00
|
|
|
async def async_update(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Get the latest date and update device state."""
|
2022-05-30 16:31:57 +00:00
|
|
|
afsapi = self.fs_device
|
|
|
|
try:
|
|
|
|
if await afsapi.get_power():
|
|
|
|
status = await afsapi.get_play_status()
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_state = {
|
2022-09-10 20:16:04 +00:00
|
|
|
PlayState.PLAYING: MediaPlayerState.PLAYING,
|
|
|
|
PlayState.PAUSED: MediaPlayerState.PAUSED,
|
|
|
|
PlayState.STOPPED: MediaPlayerState.IDLE,
|
|
|
|
PlayState.LOADING: MediaPlayerState.BUFFERING,
|
|
|
|
None: MediaPlayerState.IDLE,
|
2022-05-31 07:56:25 +00:00
|
|
|
}.get(status)
|
2022-05-30 16:31:57 +00:00
|
|
|
else:
|
2022-09-10 20:16:04 +00:00
|
|
|
self._attr_state = MediaPlayerState.OFF
|
2022-05-30 16:31:57 +00:00
|
|
|
except FSConnectionError:
|
|
|
|
if self._attr_available:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Could not connect to %s. Did it go offline?",
|
2022-05-31 09:56:44 +00:00
|
|
|
self.name or afsapi.webfsapi_endpoint,
|
2022-05-30 16:31:57 +00:00
|
|
|
)
|
|
|
|
self._attr_available = False
|
2022-05-31 07:56:25 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if not self._attr_available:
|
|
|
|
_LOGGER.info(
|
|
|
|
"Reconnected to %s",
|
2022-05-31 09:56:44 +00:00
|
|
|
self.name or afsapi.webfsapi_endpoint,
|
2022-05-31 07:56:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
self._attr_available = True
|
2022-05-31 09:56:44 +00:00
|
|
|
if not self._attr_name:
|
|
|
|
self._attr_name = await afsapi.get_friendly_name()
|
2022-05-31 07:56:25 +00:00
|
|
|
|
2022-05-31 09:56:44 +00:00
|
|
|
if not self._attr_source_list:
|
2022-05-31 07:56:25 +00:00
|
|
|
self.__modes_by_label = {
|
|
|
|
mode.label: mode.key for mode in await afsapi.get_modes()
|
|
|
|
}
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_source_list = list(self.__modes_by_label)
|
2022-05-31 07:56:25 +00:00
|
|
|
|
2022-08-23 07:41:07 +00:00
|
|
|
if not self._attr_sound_mode_list and self._supports_sound_mode:
|
|
|
|
try:
|
|
|
|
equalisers = await afsapi.get_equalisers()
|
|
|
|
except FSNotImplementedException:
|
|
|
|
self._supports_sound_mode = False
|
|
|
|
# Remove SELECT_SOUND_MODE from the advertised supported features
|
|
|
|
self._attr_supported_features ^= (
|
|
|
|
MediaPlayerEntityFeature.SELECT_SOUND_MODE
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.__sound_modes_by_label = {
|
|
|
|
sound_mode.label: sound_mode.key for sound_mode in equalisers
|
|
|
|
}
|
|
|
|
self._attr_sound_mode_list = list(self.__sound_modes_by_label)
|
2022-06-28 21:13:43 +00:00
|
|
|
|
2022-05-31 07:56:25 +00:00
|
|
|
# The API seems to include 'zero' in the number of steps (e.g. if the range is
|
|
|
|
# 0-40 then get_volume_steps returns 41) subtract one to get the max volume.
|
|
|
|
# If call to get_volume fails set to 0 and try again next time.
|
|
|
|
if not self._max_volume:
|
|
|
|
self._max_volume = int(await afsapi.get_volume_steps() or 1) - 1
|
|
|
|
|
2022-09-10 20:16:04 +00:00
|
|
|
if self._attr_state != MediaPlayerState.OFF:
|
2022-05-30 16:31:57 +00:00
|
|
|
info_name = await afsapi.get_play_name()
|
|
|
|
info_text = await afsapi.get_play_text()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_media_title = " - ".join(filter(None, [info_name, info_text]))
|
|
|
|
self._attr_media_artist = await afsapi.get_play_artist()
|
|
|
|
self._attr_media_album_name = await afsapi.get_play_album()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-07-12 10:03:26 +00:00
|
|
|
radio_mode = await afsapi.get_mode()
|
|
|
|
self._attr_source = radio_mode.label if radio_mode is not None else None
|
2022-05-31 09:56:44 +00:00
|
|
|
|
|
|
|
self._attr_is_volume_muted = await afsapi.get_mute()
|
|
|
|
self._attr_media_image_url = await afsapi.get_play_graphic()
|
2022-07-12 10:03:26 +00:00
|
|
|
|
2022-08-23 07:41:07 +00:00
|
|
|
if self._supports_sound_mode:
|
|
|
|
try:
|
|
|
|
eq_preset = await afsapi.get_eq_preset()
|
|
|
|
except FSNotImplementedException:
|
|
|
|
self._supports_sound_mode = False
|
|
|
|
# Remove SELECT_SOUND_MODE from the advertised supported features
|
|
|
|
self._attr_supported_features ^= (
|
|
|
|
MediaPlayerEntityFeature.SELECT_SOUND_MODE
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
self._attr_sound_mode = (
|
|
|
|
eq_preset.label if eq_preset is not None else None
|
|
|
|
)
|
2020-02-23 16:37:31 +00:00
|
|
|
|
|
|
|
volume = await self.fs_device.get_volume()
|
|
|
|
|
|
|
|
# Prevent division by zero if max_volume not known yet
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_volume_level = float(volume or 0) / (self._max_volume or 1)
|
2018-02-20 16:14:34 +00:00
|
|
|
else:
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_media_title = None
|
|
|
|
self._attr_media_artist = None
|
|
|
|
self._attr_media_album_name = None
|
|
|
|
|
|
|
|
self._attr_source = None
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_is_volume_muted = None
|
|
|
|
self._attr_media_image_url = None
|
2022-06-28 21:13:43 +00:00
|
|
|
self._attr_sound_mode = None
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-05-31 09:56:44 +00:00
|
|
|
self._attr_volume_level = None
|
2020-02-23 16:37:31 +00:00
|
|
|
|
2018-02-20 16:14:34 +00:00
|
|
|
# Management actions
|
2017-02-28 14:23:07 +00:00
|
|
|
# power control
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_turn_on(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Turn on the device."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.set_power(True)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_turn_off(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Turn off the device."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.set_power(False)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_media_play(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send play command."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.play()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_media_pause(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send pause command."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.pause()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_media_play_pause(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send play/pause command."""
|
2022-09-10 20:16:04 +00:00
|
|
|
if self._attr_state == MediaPlayerState.PLAYING:
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.pause()
|
2017-02-28 14:23:07 +00:00
|
|
|
else:
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.play()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_media_stop(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send play/pause command."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.pause()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_media_previous_track(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send previous track command (results in rewind)."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.rewind()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_media_next_track(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send next track command (results in fast-forward)."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.forward()
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_mute_volume(self, mute: bool) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send mute command."""
|
2018-10-01 06:58:21 +00:00
|
|
|
await self.fs_device.set_mute(mute)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
|
|
|
# volume
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_volume_up(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send volume up command."""
|
2018-10-01 06:58:21 +00:00
|
|
|
volume = await self.fs_device.get_volume()
|
2020-02-23 16:37:31 +00:00
|
|
|
volume = int(volume or 0) + 1
|
|
|
|
await self.fs_device.set_volume(min(volume, self._max_volume))
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_volume_down(self) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Send volume down command."""
|
2018-10-01 06:58:21 +00:00
|
|
|
volume = await self.fs_device.get_volume()
|
2020-02-23 16:37:31 +00:00
|
|
|
volume = int(volume or 0) - 1
|
|
|
|
await self.fs_device.set_volume(max(volume, 0))
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-22 11:36:33 +00:00
|
|
|
async def async_set_volume_level(self, volume: float) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Set volume command."""
|
2020-02-23 16:37:31 +00:00
|
|
|
if self._max_volume: # Can't do anything sensible if not set
|
|
|
|
volume = int(volume * self._max_volume)
|
|
|
|
await self.fs_device.set_volume(volume)
|
2017-02-28 14:23:07 +00:00
|
|
|
|
2022-08-26 08:54:22 +00:00
|
|
|
async def async_select_source(self, source: str) -> None:
|
2017-02-28 14:23:07 +00:00
|
|
|
"""Select input source."""
|
2022-05-30 16:31:57 +00:00
|
|
|
await self.fs_device.set_power(True)
|
2022-08-26 08:54:22 +00:00
|
|
|
if (
|
|
|
|
self.__modes_by_label
|
|
|
|
and (mode := self.__modes_by_label.get(source)) is not None
|
|
|
|
):
|
|
|
|
await self.fs_device.set_mode(mode)
|
2022-06-28 21:13:43 +00:00
|
|
|
|
2022-08-26 08:54:22 +00:00
|
|
|
async def async_select_sound_mode(self, sound_mode: str) -> None:
|
2022-06-28 21:13:43 +00:00
|
|
|
"""Select EQ Preset."""
|
2022-08-26 08:54:22 +00:00
|
|
|
if (
|
|
|
|
self.__sound_modes_by_label
|
|
|
|
and (mode := self.__sound_modes_by_label.get(sound_mode)) is not None
|
|
|
|
):
|
|
|
|
await self.fs_device.set_eq_preset(mode)
|
2022-11-29 12:31:49 +00:00
|
|
|
|
|
|
|
async def async_browse_media(
|
|
|
|
self, media_content_type: str | None = None, media_content_id: str | None = None
|
|
|
|
) -> BrowseMedia:
|
|
|
|
"""Browse media library and preset stations."""
|
|
|
|
if not media_content_id:
|
|
|
|
return await browse_top_level(self._attr_source, self.fs_device)
|
|
|
|
|
|
|
|
return await browse_node(self.fs_device, media_content_type, media_content_id)
|
|
|
|
|
|
|
|
async def async_play_media(
|
|
|
|
self, media_type: MediaType | str, media_id: str, **kwargs: Any
|
|
|
|
) -> None:
|
|
|
|
"""Play selected media or channel."""
|
|
|
|
if media_type != MediaType.CHANNEL:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Got %s, but frontier_silicon only supports playing channels",
|
|
|
|
media_type,
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
player_mode, media_type, *keys = media_id.split("/")
|
|
|
|
|
|
|
|
await self.async_select_source(player_mode) # this also powers on the device
|
|
|
|
|
|
|
|
if media_type == MEDIA_CONTENT_ID_PRESET:
|
|
|
|
if len(keys) != 1:
|
|
|
|
raise BrowseError("Presets can only have 1 level")
|
|
|
|
|
|
|
|
# Keys of presets are 0-based, while the list shown on the device starts from 1
|
|
|
|
preset = int(keys[0]) - 1
|
|
|
|
|
|
|
|
result = await self.fs_device.select_preset(preset)
|
|
|
|
else:
|
|
|
|
result = await self.fs_device.nav_select_item_via_path(keys)
|
|
|
|
|
|
|
|
await self.async_update()
|
|
|
|
self._attr_media_content_id = media_id
|
|
|
|
return result
|