Module pyatv.interface
Public interface exposed by library.
This module contains all the interfaces that represents a generic Apple TV device and all its features.
Expand source code
"""Public interface exposed by library.
This module contains all the interfaces that represents a generic Apple TV device and
all its features.
"""
import re
import inspect
import hashlib
from typing import Dict, Optional, NamedTuple
from abc import ABC, abstractmethod
import aiohttp
from pyatv import const, convert, exceptions
from pyatv.const import (
Protocol,
OperatingSystem,
DeviceModel,
FeatureState,
FeatureName,
)
from pyatv.support import net
_ALL_FEATURES = {} # type: Dict[int, str]
class ArtworkInfo(NamedTuple):
"""Artwork information."""
bytes: bytes
mimetype: str
class FeatureInfo(NamedTuple):
"""Feature state and options."""
state: FeatureState
options: Optional[Dict[str, object]] = {}
def feature(index, name, doc):
"""Decorate functions and properties as a feature.
Note: This is an internal function.
"""
def _wraps(obj):
if index in _ALL_FEATURES:
raise Exception(
f"Index {index} collides between {name} and {_ALL_FEATURES[index]}"
)
_ALL_FEATURES[index] = (name, doc)
return obj
return _wraps
def _get_first_sentence_in_pydoc(obj):
doc = obj.__doc__
index = doc.find(".")
if index == -1:
# Here we have no leading . so return everything
return doc
# Try to find the first complete sentence and respect
# abbreviations correctly
match = re.findall(r"(.*\.[^A-Z]*)\.(?: [A-Z].*|)", doc)
if len(match) == 1:
return match[0]
return doc[0:index]
def retrieve_commands(obj: object):
"""Retrieve all commands and help texts from an API object."""
commands = {} # type: Dict[str, str]
for func in obj.__dict__:
if not inspect.isfunction(obj.__dict__[func]) and not isinstance(
obj.__dict__[func], property
):
continue
if func.startswith("_") or func == "listener":
continue
commands[func] = _get_first_sentence_in_pydoc(obj.__dict__[func])
return commands
class BaseService:
"""Base class for protocol services."""
def __init__(
self,
identifier: str,
protocol: Protocol,
port: int,
properties: Optional[Dict[str, str]],
) -> None:
"""Initialize a new BaseService."""
self.__identifier = identifier
self.protocol = protocol
self.port = port
self.credentials = None # type: Optional[str]
self.properties = properties or {}
@property
def identifier(self) -> str:
"""Return unique identifier associated with this service."""
return self.__identifier
def merge(self, other) -> None:
"""Merge with other service of same type."""
self.credentials = other.credentials or self.credentials
self.properties.update(other.properties)
def __str__(self) -> str:
"""Return a string representation of this object."""
return "Protocol: {0}, Port: {1}, Credentials: {2}".format(
convert.protocol_str(self.protocol), self.port, self.credentials
)
class PairingHandler(ABC):
"""Base class for API used to pair with an Apple TV."""
def __init__(self, session: aiohttp.ClientSession, service: BaseService) -> None:
"""Initialize a new instance of PairingHandler."""
self.session = session
self._service = service
@property
def service(self) -> BaseService:
"""Return service used for pairing."""
return self._service
async def close(self) -> None:
"""Call to free allocated resources after pairing."""
if net.is_custom_session(self.session):
await self.session.close()
@abstractmethod
def pin(self, pin) -> None:
"""Pin code used for pairing."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def device_provides_pin(self) -> bool:
"""Return True if remote device presents PIN code, else False."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def has_paired(self) -> bool:
"""If a successful pairing has been performed.
The value will be reset when stop() is called.
"""
raise exceptions.NotSupportedError()
@abstractmethod
async def begin(self) -> None:
"""Start pairing process."""
raise exceptions.NotSupportedError()
@abstractmethod
async def finish(self) -> None:
"""Stop pairing process."""
raise exceptions.NotSupportedError()
class RemoteControl(ABC): # pylint: disable=too-many-public-methods
"""Base class for API used to control an Apple TV."""
# pylint: disable=invalid-name
@abstractmethod
@feature(0, "Up", "Up button on remote.")
def up(self) -> None:
"""Press key up."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(1, "Down", "Down button on remote.")
def down(self) -> None:
"""Press key down."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(2, "Left", "Left button on remote.")
def left(self) -> None:
"""Press key left."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(3, "Right", "Right button on remote.")
def right(self) -> None:
"""Press key right."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(4, "Play", "Start playing media.")
def play(self) -> None:
"""Press key play."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(5, "PlayPause", "Toggle between play/pause.")
def play_pause(self) -> None:
"""Toggle between play and pause."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(6, "Pause", "Pause playing media.")
def pause(self) -> None:
"""Press key play."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(7, "Stop", "Stop playing media.")
def stop(self) -> None:
"""Press key stop."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(8, "Next", "Change to next item.")
def next(self) -> None:
"""Press key next."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(9, "Previous", "Change to previous item.")
def previous(self) -> None:
"""Press key previous."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(10, "Select", "Select current option.")
def select(self) -> None:
"""Press key select."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(11, "Menu", "Go back to previos menu.")
def menu(self) -> None:
"""Press key menu."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(12, "VolumeUp", "Increase volume.")
def volume_up(self) -> None:
"""Press key volume up."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(13, "VolumeDown", "Decrease volume.")
def volume_down(self) -> None:
"""Press key volume down."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(14, "Home", "Home/TV button.")
def home(self) -> None:
"""Press key home."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(15, "HomeHold", "Long-press home button.")
def home_hold(self) -> None:
"""Hold key home."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(16, "TopMenu", "Go to main menu.")
def top_menu(self) -> None:
"""Go to main menu (long press menu)."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(17, "Suspend", "Suspend device (deprecated; use Power.turn_off).")
def suspend(self) -> None:
"""Suspend the device."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(18, "WakeUp", "Wake up device (deprecated; use Power.turn_on).")
def wakeup(self) -> None:
"""Wake up the device."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(19, "SetPosition", "Seek to position.")
def set_position(self, pos) -> None:
"""Seek in the current playing media."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(20, "SetShuffle", "Change shuffle state.")
def set_shuffle(self, shuffle_state) -> None:
"""Change shuffle mode to on or off."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(21, "SetRepeat", "Change repeat state.")
def set_repeat(self, repeat_state) -> None:
"""Change repeat state."""
raise exceptions.NotSupportedError()
class Playing(ABC):
"""Base class for retrieving what is currently playing."""
def __str__(self) -> str:
"""Convert this playing object to a readable string."""
output = []
output.append(
" Media type: {0}".format(convert.media_type_str(self.media_type))
)
output.append(
"Device state: {0}".format(convert.device_state_str(self.device_state))
)
if self.title is not None:
output.append(" Title: {0}".format(self.title))
if self.artist is not None:
output.append(" Artist: {0}".format(self.artist))
if self.album is not None:
output.append(" Album: {0}".format(self.album))
if self.genre is not None:
output.append(" Genre: {0}".format(self.genre))
position = self.position
total_time = self.total_time
if position is not None and total_time is not None and total_time != 0:
output.append(
" Position: {0}/{1}s ({2:.1%})".format(
position, total_time, float(position) / float(total_time)
)
)
elif position is not None and position != 0:
output.append(" Position: {0}s".format(position))
elif total_time is not None and position != 0:
output.append(" Total time: {0}s".format(total_time))
if self.repeat is not None:
output.append(" Repeat: {0}".format(convert.repeat_str(self.repeat)))
if self.shuffle is not None:
output.append(" Shuffle: {0}".format(convert.shuffle_str(self.shuffle)))
return "\n".join(output)
@property
def hash(self) -> str:
"""Create a unique hash for what is currently playing.
The hash is based on title, artist, album and total time. It should
always be the same for the same content, but it is not guaranteed.
"""
base = "{0}{1}{2}{3}".format(
self.title, self.artist, self.album, self.total_time
)
return hashlib.sha256(base.encode("utf-8")).hexdigest()
@property
@abstractmethod
def media_type(self) -> const.MediaType:
"""Type of media is currently playing, e.g. video, music."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def device_state(self) -> const.DeviceState:
"""Device state, e.g. playing or paused."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(22, "Title", "Title of playing media.")
def title(self) -> Optional[str]:
"""Title of the current media, e.g. movie or song name."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(23, "Artist", "Artist of playing song.")
def artist(self) -> Optional[str]:
"""Artist of the currently playing song."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(24, "Album", "Album from playing artist.")
def album(self) -> Optional[str]:
"""Album of the currently playing song."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(25, "Genre", "Genre of playing song.")
def genre(self) -> Optional[str]:
"""Genre of the currently playing song."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(26, "TotalTime", "Total length of playing media (seconds).")
def total_time(self) -> int:
"""Total play time in seconds."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(27, "Position", "Current play time position.")
def position(self) -> int:
"""Position in the playing media (seconds)."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(28, "Shuffle", "Shuffle stat.e")
def shuffle(self) -> const.ShuffleState:
"""If shuffle is enabled or not."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(29, "Repeat", "Repeat state.")
def repeat(self) -> const.RepeatState:
"""Repeat mode."""
raise exceptions.NotSupportedError()
class App:
"""Information about an app."""
def __init__(self, name: Optional[str], identifier: str) -> None:
"""Initialize a new App instance."""
self._name = name
self._identifier = identifier
@property
def name(self) -> Optional[str]:
"""User friendly name of app."""
return self._name
@property
def identifier(self) -> str:
"""Return a unique bundle id for the app."""
return self._identifier
def __str__(self) -> str:
"""Convert app info to readable string."""
return f"App: {self.name} ({self.identifier})"
class Metadata(ABC):
"""Base class for retrieving metadata from an Apple TV."""
def __init__(self, identifier: str) -> None:
"""Initialize a new instance of Metadata."""
self._identifier = identifier
@property
def device_id(self) -> Optional[str]:
"""Return a unique identifier for current device."""
return self._identifier
@abstractmethod
@feature(30, "Artwork", "Playing media artwork.")
def artwork(self) -> Optional[ArtworkInfo]:
"""Return artwork for what is currently playing (or None)."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def artwork_id(self) -> str:
"""Return a unique identifier for current artwork."""
raise exceptions.NotSupportedError()
@abstractmethod
def playing(self) -> Playing:
"""Return what is currently playing."""
raise exceptions.NotSupportedError()
@property # type: ignore
@abstractmethod
@feature(35, "App", "App playing media.")
def app(self) -> Optional[App]:
"""Return information about current app playing something.
Do note that this property returns which app is currently playing something and
not which app is currently active. If nothing is playing, the corresponding
feature will be unavailable.
"""
raise exceptions.NotSupportedError()
class PushListener(ABC):
"""Listener interface for push updates."""
@abstractmethod
def playstatus_update(self, updater, playstatus: Playing) -> None:
"""Inform about changes to what is currently playing."""
@abstractmethod
def playstatus_error(self, updater, exception: Exception) -> None:
"""Inform about an error when updating play status."""
class PushUpdater(ABC):
"""Base class for push/async updates from an Apple TV."""
def __init__(self) -> None:
"""Initialize a new PushUpdater."""
self.__listener: Optional[PushListener] = None
@property
@abstractmethod
def active(self) -> bool:
"""Return if push updater has been started."""
raise exceptions.NotSupportedError()
@property
def listener(self) -> Optional[PushListener]:
"""Object (PushUpdaterListener) that receives updates."""
return self.__listener
@listener.setter
def listener(self, listener: Optional[PushListener]) -> None:
"""Object that receives updates.
This should be an object implementing two methods:
- playstatus_update(updater, playstatus)
- playstatus_error(updater, exception)
The first method is called when a new update happens and the second one
is called if an error occurs.
"""
self.__listener = listener
@abstractmethod
def start(self, initial_delay: int = 0) -> None:
"""Begin to listen to updates.
If an error occurs, start must be called again.
"""
raise exceptions.NotSupportedError()
@abstractmethod
def stop(self) -> None:
"""No longer forward updates to listener."""
raise exceptions.NotSupportedError()
class Stream(ABC): # pylint: disable=too-few-public-methods
"""Base class for stream functionality."""
@abstractmethod
@feature(31, "PlayUrl", "Stream a URL on device.")
def play_url(self, url: str, **kwargs) -> None:
"""Play media from an URL on the device."""
raise exceptions.NotSupportedError()
class DeviceListener(ABC):
"""Listener interface for generic device updates."""
@abstractmethod
def connection_lost(self, exception: Exception) -> None:
"""Device was unexpectedly disconnected."""
raise NotImplementedError()
@abstractmethod
def connection_closed(self) -> None:
"""Device connection was (intentionally) closed."""
raise NotImplementedError()
class PowerListener(ABC): # pylint: disable=too-few-public-methods
"""Listener interface for power updates."""
@abstractmethod
def powerstate_update(
self, old_state: const.PowerState, new_state: const.PowerState
):
"""Device power state was updated."""
raise NotImplementedError()
class Power(ABC):
"""Base class for retrieving power state from an Apple TV."""
def __init__(self) -> None:
"""Initialize a new Power instance."""
self.__listener: Optional[PowerListener] = None
@property
def listener(self) -> Optional[PowerListener]:
"""Object receiving power state updates.
Must be an object conforming to PowerListener.
"""
return self.__listener
@listener.setter
def listener(self, listener: Optional[PowerListener]) -> None:
"""Object that receives updates.
This should be an object implementing method:
- powerstate_update(old_state, new_state)
"""
self.__listener = listener
@property # type: ignore
@abstractmethod
@feature(32, "PowerState", "Current device power state.")
def power_state(self) -> const.PowerState:
"""Return device power state."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(33, "TurnOn", "Turn device on.")
async def turn_on(self) -> None:
"""Turn device on."""
raise exceptions.NotSupportedError()
@abstractmethod
@feature(34, "TurnOff", "Turn off device.")
async def turn_off(self) -> None:
"""Turn device off."""
raise exceptions.NotSupportedError()
class DeviceInfo:
"""General information about device."""
def __init__(
self,
os: const.OperatingSystem,
version: Optional[str],
build_number: Optional[str],
model: const.DeviceModel,
mac: Optional[str],
) -> None: # pylint: disable=too-many-arguments # noqa
"""Initialize a new DeviceInfo instance."""
self._os = os
self._version = version
self._build_number = build_number
self._model = model
self._mac = mac
@property
def operating_system(self) -> const.OperatingSystem:
"""Operating system running on device."""
return self._os
@property
def version(self) -> Optional[str]:
"""Operating system version."""
return self._version
@property
def build_number(self) -> Optional[str]:
"""Operating system build number, e.g. 17K795."""
return self._build_number
@property
def model(self) -> const.DeviceModel:
"""Hardware model name, e.g. 3, 4 or 4K."""
return self._model
@property
def mac(self) -> Optional[str]:
"""Device MAC address."""
return self._mac
def __str__(self) -> str:
"""Convert device info to readable string."""
if self.model != DeviceModel.Unknown:
output = self.model.name.replace("Gen", "")
else:
output = "Unknown Model"
output += " " + {
OperatingSystem.Legacy: "ATV SW",
OperatingSystem.TvOS: "tvOS",
}.get(self.operating_system, "Unknown OS")
if self.version:
output += " " + self.version
if self.build_number:
output += " build " + self.build_number
return output
class Features(ABC):
"""Base class for supported feature functionality."""
@abstractmethod
def get_feature(self, feature: FeatureName) -> FeatureInfo:
"""Return current state of a feature."""
raise NotImplementedError()
def all_features(self, include_unsupported=False) -> Dict[FeatureName, FeatureInfo]:
"""Return state of all features."""
features = {} # type: Dict[FeatureName, FeatureInfo]
for name in FeatureName:
feature = self.get_feature(name)
if feature.state != FeatureState.Unsupported or include_unsupported:
features[name] = feature
return features
class AppleTV(ABC):
"""Base class representing an Apple TV."""
def __init__(self) -> None:
"""Initialize a new AppleTV."""
self.__listener: Optional[DeviceListener] = None
@property
def listener(self) -> Optional[DeviceListener]:
"""Object receiving generic device updates.
Must be an object conforming to DeviceListener.
"""
return self.__listener
@listener.setter
def listener(self, target: Optional[DeviceListener]):
"""Change object receiving generic device updates."""
self.__listener = target
@abstractmethod
async def connect(self) -> None:
"""Initiate connection to device.
No need to call it yourself, it's done automatically.
"""
raise exceptions.NotSupportedError()
@abstractmethod
async def close(self) -> None:
"""Close connection and release allocated resources."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def device_info(self) -> DeviceInfo:
"""Return API for device information."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def service(self) -> BaseService:
"""Return service used to connect to the Apple TV."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def remote_control(self) -> RemoteControl:
"""Return API for controlling the Apple TV."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def metadata(self) -> Metadata:
"""Return API for retrieving metadata from the Apple TV."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def push_updater(self) -> PushUpdater:
"""Return API for handling push update from the Apple TV."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def stream(self) -> Stream:
"""Return API for streaming media."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def power(self) -> Power:
"""Return API for power management."""
raise exceptions.NotSupportedError()
@property
@abstractmethod
def features(self) -> Features:
"""Return features interface."""
raise exceptions.NotSupportedError()
Functions
def feature(index, name, doc)
-
Decorate functions and properties as a feature.
Note: This is an internal function.
Expand source code
def feature(index, name, doc): """Decorate functions and properties as a feature. Note: This is an internal function. """ def _wraps(obj): if index in _ALL_FEATURES: raise Exception( f"Index {index} collides between {name} and {_ALL_FEATURES[index]}" ) _ALL_FEATURES[index] = (name, doc) return obj return _wraps
def retrieve_commands(obj)
-
Retrieve all commands and help texts from an API object.
Expand source code
def retrieve_commands(obj: object): """Retrieve all commands and help texts from an API object.""" commands = {} # type: Dict[str, str] for func in obj.__dict__: if not inspect.isfunction(obj.__dict__[func]) and not isinstance( obj.__dict__[func], property ): continue if func.startswith("_") or func == "listener": continue commands[func] = _get_first_sentence_in_pydoc(obj.__dict__[func]) return commands
Classes
class App (name, identifier)
-
Information about an app.
Initialize a new App instance.
Expand source code
class App: """Information about an app.""" def __init__(self, name: Optional[str], identifier: str) -> None: """Initialize a new App instance.""" self._name = name self._identifier = identifier @property def name(self) -> Optional[str]: """User friendly name of app.""" return self._name @property def identifier(self) -> str: """Return a unique bundle id for the app.""" return self._identifier def __str__(self) -> str: """Convert app info to readable string.""" return f"App: {self.name} ({self.identifier})"
Instance variables
var identifier
-
Return a unique bundle id for the app.
Expand source code
@property def identifier(self) -> str: """Return a unique bundle id for the app.""" return self._identifier
var name
-
User friendly name of app.
Expand source code
@property def name(self) -> Optional[str]: """User friendly name of app.""" return self._name
class AppleTV
-
Base class representing an Apple TV.
Initialize a new AppleTV.
Expand source code
class AppleTV(ABC): """Base class representing an Apple TV.""" def __init__(self) -> None: """Initialize a new AppleTV.""" self.__listener: Optional[DeviceListener] = None @property def listener(self) -> Optional[DeviceListener]: """Object receiving generic device updates. Must be an object conforming to DeviceListener. """ return self.__listener @listener.setter def listener(self, target: Optional[DeviceListener]): """Change object receiving generic device updates.""" self.__listener = target @abstractmethod async def connect(self) -> None: """Initiate connection to device. No need to call it yourself, it's done automatically. """ raise exceptions.NotSupportedError() @abstractmethod async def close(self) -> None: """Close connection and release allocated resources.""" raise exceptions.NotSupportedError() @property @abstractmethod def device_info(self) -> DeviceInfo: """Return API for device information.""" raise exceptions.NotSupportedError() @property @abstractmethod def service(self) -> BaseService: """Return service used to connect to the Apple TV.""" raise exceptions.NotSupportedError() @property @abstractmethod def remote_control(self) -> RemoteControl: """Return API for controlling the Apple TV.""" raise exceptions.NotSupportedError() @property @abstractmethod def metadata(self) -> Metadata: """Return API for retrieving metadata from the Apple TV.""" raise exceptions.NotSupportedError() @property @abstractmethod def push_updater(self) -> PushUpdater: """Return API for handling push update from the Apple TV.""" raise exceptions.NotSupportedError() @property @abstractmethod def stream(self) -> Stream: """Return API for streaming media.""" raise exceptions.NotSupportedError() @property @abstractmethod def power(self) -> Power: """Return API for power management.""" raise exceptions.NotSupportedError() @property @abstractmethod def features(self) -> Features: """Return features interface.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapAppleTV
- pyatv.mrp.MrpAppleTV
Instance variables
var device_info
-
Return API for device information.
Expand source code
@property @abstractmethod def device_info(self) -> DeviceInfo: """Return API for device information.""" raise exceptions.NotSupportedError()
var features
-
Return features interface.
Expand source code
@property @abstractmethod def features(self) -> Features: """Return features interface.""" raise exceptions.NotSupportedError()
var listener
-
Object receiving generic device updates.
Must be an object conforming to DeviceListener.
Expand source code
@property def listener(self) -> Optional[DeviceListener]: """Object receiving generic device updates. Must be an object conforming to DeviceListener. """ return self.__listener
var metadata
-
Return API for retrieving metadata from the Apple TV.
Expand source code
@property @abstractmethod def metadata(self) -> Metadata: """Return API for retrieving metadata from the Apple TV.""" raise exceptions.NotSupportedError()
var power
-
Return API for power management.
Expand source code
@property @abstractmethod def power(self) -> Power: """Return API for power management.""" raise exceptions.NotSupportedError()
var push_updater
-
Return API for handling push update from the Apple TV.
Expand source code
@property @abstractmethod def push_updater(self) -> PushUpdater: """Return API for handling push update from the Apple TV.""" raise exceptions.NotSupportedError()
var remote_control
-
Return API for controlling the Apple TV.
Expand source code
@property @abstractmethod def remote_control(self) -> RemoteControl: """Return API for controlling the Apple TV.""" raise exceptions.NotSupportedError()
var service
-
Return service used to connect to the Apple TV.
Expand source code
@property @abstractmethod def service(self) -> BaseService: """Return service used to connect to the Apple TV.""" raise exceptions.NotSupportedError()
var stream
-
Return API for streaming media.
Expand source code
@property @abstractmethod def stream(self) -> Stream: """Return API for streaming media.""" raise exceptions.NotSupportedError()
Methods
async def close(self)
-
Close connection and release allocated resources.
Expand source code
@abstractmethod async def close(self) -> None: """Close connection and release allocated resources.""" raise exceptions.NotSupportedError()
async def connect(self)
-
Initiate connection to device.
No need to call it yourself, it's done automatically.
Expand source code
@abstractmethod async def connect(self) -> None: """Initiate connection to device. No need to call it yourself, it's done automatically. """ raise exceptions.NotSupportedError()
class ArtworkInfo (bytes, mimetype)
-
Artwork information.
Expand source code
class ArtworkInfo(NamedTuple): """Artwork information.""" bytes: bytes mimetype: str
Ancestors
- builtins.tuple
Instance variables
var bytes
-
Alias for field number 0
var mimetype
-
Alias for field number 1
class BaseService (identifier, protocol, port, properties)
-
Base class for protocol services.
Initialize a new BaseService.
Expand source code
class BaseService: """Base class for protocol services.""" def __init__( self, identifier: str, protocol: Protocol, port: int, properties: Optional[Dict[str, str]], ) -> None: """Initialize a new BaseService.""" self.__identifier = identifier self.protocol = protocol self.port = port self.credentials = None # type: Optional[str] self.properties = properties or {} @property def identifier(self) -> str: """Return unique identifier associated with this service.""" return self.__identifier def merge(self, other) -> None: """Merge with other service of same type.""" self.credentials = other.credentials or self.credentials self.properties.update(other.properties) def __str__(self) -> str: """Return a string representation of this object.""" return "Protocol: {0}, Port: {1}, Credentials: {2}".format( convert.protocol_str(self.protocol), self.port, self.credentials )
Subclasses
Instance variables
var identifier
-
Return unique identifier associated with this service.
Expand source code
@property def identifier(self) -> str: """Return unique identifier associated with this service.""" return self.__identifier
Methods
def merge(self, other)
-
Merge with other service of same type.
Expand source code
def merge(self, other) -> None: """Merge with other service of same type.""" self.credentials = other.credentials or self.credentials self.properties.update(other.properties)
class DeviceInfo (os, version, build_number, model, mac)
-
General information about device.
Initialize a new DeviceInfo instance.
Expand source code
class DeviceInfo: """General information about device.""" def __init__( self, os: const.OperatingSystem, version: Optional[str], build_number: Optional[str], model: const.DeviceModel, mac: Optional[str], ) -> None: # pylint: disable=too-many-arguments # noqa """Initialize a new DeviceInfo instance.""" self._os = os self._version = version self._build_number = build_number self._model = model self._mac = mac @property def operating_system(self) -> const.OperatingSystem: """Operating system running on device.""" return self._os @property def version(self) -> Optional[str]: """Operating system version.""" return self._version @property def build_number(self) -> Optional[str]: """Operating system build number, e.g. 17K795.""" return self._build_number @property def model(self) -> const.DeviceModel: """Hardware model name, e.g. 3, 4 or 4K.""" return self._model @property def mac(self) -> Optional[str]: """Device MAC address.""" return self._mac def __str__(self) -> str: """Convert device info to readable string.""" if self.model != DeviceModel.Unknown: output = self.model.name.replace("Gen", "") else: output = "Unknown Model" output += " " + { OperatingSystem.Legacy: "ATV SW", OperatingSystem.TvOS: "tvOS", }.get(self.operating_system, "Unknown OS") if self.version: output += " " + self.version if self.build_number: output += " build " + self.build_number return output
Instance variables
var build_number
-
Operating system build number, e.g. 17K795.
Expand source code
@property def build_number(self) -> Optional[str]: """Operating system build number, e.g. 17K795.""" return self._build_number
var mac
-
Device MAC address.
Expand source code
@property def mac(self) -> Optional[str]: """Device MAC address.""" return self._mac
var model
-
Hardware model name, e.g. 3, 4 or 4K.
Expand source code
@property def model(self) -> const.DeviceModel: """Hardware model name, e.g. 3, 4 or 4K.""" return self._model
var operating_system
-
Operating system running on device.
Expand source code
@property def operating_system(self) -> const.OperatingSystem: """Operating system running on device.""" return self._os
var version
-
Operating system version.
Expand source code
@property def version(self) -> Optional[str]: """Operating system version.""" return self._version
class DeviceListener
-
Listener interface for generic device updates.
Expand source code
class DeviceListener(ABC): """Listener interface for generic device updates.""" @abstractmethod def connection_lost(self, exception: Exception) -> None: """Device was unexpectedly disconnected.""" raise NotImplementedError() @abstractmethod def connection_closed(self) -> None: """Device connection was (intentionally) closed.""" raise NotImplementedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.scripts.atvremote.DeviceListener
Methods
def connection_closed(self)
-
Device connection was (intentionally) closed.
Expand source code
@abstractmethod def connection_closed(self) -> None: """Device connection was (intentionally) closed.""" raise NotImplementedError()
def connection_lost(self, exception)
-
Device was unexpectedly disconnected.
Expand source code
@abstractmethod def connection_lost(self, exception: Exception) -> None: """Device was unexpectedly disconnected.""" raise NotImplementedError()
class FeatureInfo (state, options={})
-
Feature state and options.
Expand source code
class FeatureInfo(NamedTuple): """Feature state and options.""" state: FeatureState options: Optional[Dict[str, object]] = {}
Ancestors
- builtins.tuple
Instance variables
var options
-
Alias for field number 1
var state
-
Alias for field number 0
class Features
-
Base class for supported feature functionality.
Expand source code
class Features(ABC): """Base class for supported feature functionality.""" @abstractmethod def get_feature(self, feature: FeatureName) -> FeatureInfo: """Return current state of a feature.""" raise NotImplementedError() def all_features(self, include_unsupported=False) -> Dict[FeatureName, FeatureInfo]: """Return state of all features.""" features = {} # type: Dict[FeatureName, FeatureInfo] for name in FeatureName: feature = self.get_feature(name) if feature.state != FeatureState.Unsupported or include_unsupported: features[name] = feature return features
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapFeatures
- pyatv.mrp.MrpFeatures
Methods
def all_features(self, include_unsupported=False)
-
Return state of all features.
Expand source code
def all_features(self, include_unsupported=False) -> Dict[FeatureName, FeatureInfo]: """Return state of all features.""" features = {} # type: Dict[FeatureName, FeatureInfo] for name in FeatureName: feature = self.get_feature(name) if feature.state != FeatureState.Unsupported or include_unsupported: features[name] = feature return features
def get_feature(self, feature)
-
Return current state of a feature.
Expand source code
@abstractmethod def get_feature(self, feature: FeatureName) -> FeatureInfo: """Return current state of a feature.""" raise NotImplementedError()
class Metadata (identifier)
-
Base class for retrieving metadata from an Apple TV.
Initialize a new instance of Metadata.
Expand source code
class Metadata(ABC): """Base class for retrieving metadata from an Apple TV.""" def __init__(self, identifier: str) -> None: """Initialize a new instance of Metadata.""" self._identifier = identifier @property def device_id(self) -> Optional[str]: """Return a unique identifier for current device.""" return self._identifier @abstractmethod @feature(30, "Artwork", "Playing media artwork.") def artwork(self) -> Optional[ArtworkInfo]: """Return artwork for what is currently playing (or None).""" raise exceptions.NotSupportedError() @property @abstractmethod def artwork_id(self) -> str: """Return a unique identifier for current artwork.""" raise exceptions.NotSupportedError() @abstractmethod def playing(self) -> Playing: """Return what is currently playing.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(35, "App", "App playing media.") def app(self) -> Optional[App]: """Return information about current app playing something. Do note that this property returns which app is currently playing something and not which app is currently active. If nothing is playing, the corresponding feature will be unavailable. """ raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapMetadata
- pyatv.mrp.MrpMetadata
Instance variables
var app
-
Return information about current app playing something.
Do note that this property returns which app is currently playing something and not which app is currently active. If nothing is playing, the corresponding feature will be unavailable.
Expand source code
@property # type: ignore @abstractmethod @feature(35, "App", "App playing media.") def app(self) -> Optional[App]: """Return information about current app playing something. Do note that this property returns which app is currently playing something and not which app is currently active. If nothing is playing, the corresponding feature will be unavailable. """ raise exceptions.NotSupportedError()
var artwork_id
-
Return a unique identifier for current artwork.
Expand source code
@property @abstractmethod def artwork_id(self) -> str: """Return a unique identifier for current artwork.""" raise exceptions.NotSupportedError()
var device_id
-
Return a unique identifier for current device.
Expand source code
@property def device_id(self) -> Optional[str]: """Return a unique identifier for current device.""" return self._identifier
Methods
def artwork(self)
-
Return artwork for what is currently playing (or None).
Expand source code
@abstractmethod @feature(30, "Artwork", "Playing media artwork.") def artwork(self) -> Optional[ArtworkInfo]: """Return artwork for what is currently playing (or None).""" raise exceptions.NotSupportedError()
def playing(self)
-
Return what is currently playing.
Expand source code
@abstractmethod def playing(self) -> Playing: """Return what is currently playing.""" raise exceptions.NotSupportedError()
class PairingHandler (session, service)
-
Base class for API used to pair with an Apple TV.
Initialize a new instance of PairingHandler.
Expand source code
class PairingHandler(ABC): """Base class for API used to pair with an Apple TV.""" def __init__(self, session: aiohttp.ClientSession, service: BaseService) -> None: """Initialize a new instance of PairingHandler.""" self.session = session self._service = service @property def service(self) -> BaseService: """Return service used for pairing.""" return self._service async def close(self) -> None: """Call to free allocated resources after pairing.""" if net.is_custom_session(self.session): await self.session.close() @abstractmethod def pin(self, pin) -> None: """Pin code used for pairing.""" raise exceptions.NotSupportedError() @property @abstractmethod def device_provides_pin(self) -> bool: """Return True if remote device presents PIN code, else False.""" raise exceptions.NotSupportedError() @property @abstractmethod def has_paired(self) -> bool: """If a successful pairing has been performed. The value will be reset when stop() is called. """ raise exceptions.NotSupportedError() @abstractmethod async def begin(self) -> None: """Start pairing process.""" raise exceptions.NotSupportedError() @abstractmethod async def finish(self) -> None: """Stop pairing process.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.airplay.pairing.AirPlayPairingHandler
- pyatv.dmap.pairing.DmapPairingHandler
- pyatv.mrp.pairing.MrpPairingHandler
Instance variables
var device_provides_pin
-
Return True if remote device presents PIN code, else False.
Expand source code
@property @abstractmethod def device_provides_pin(self) -> bool: """Return True if remote device presents PIN code, else False.""" raise exceptions.NotSupportedError()
var has_paired
-
If a successful pairing has been performed.
The value will be reset when stop() is called.
Expand source code
@property @abstractmethod def has_paired(self) -> bool: """If a successful pairing has been performed. The value will be reset when stop() is called. """ raise exceptions.NotSupportedError()
var service
-
Return service used for pairing.
Expand source code
@property def service(self) -> BaseService: """Return service used for pairing.""" return self._service
Methods
async def begin(self)
-
Start pairing process.
Expand source code
@abstractmethod async def begin(self) -> None: """Start pairing process.""" raise exceptions.NotSupportedError()
async def close(self)
-
Call to free allocated resources after pairing.
Expand source code
async def close(self) -> None: """Call to free allocated resources after pairing.""" if net.is_custom_session(self.session): await self.session.close()
async def finish(self)
-
Stop pairing process.
Expand source code
@abstractmethod async def finish(self) -> None: """Stop pairing process.""" raise exceptions.NotSupportedError()
def pin(self, pin)
-
Pin code used for pairing.
Expand source code
@abstractmethod def pin(self, pin) -> None: """Pin code used for pairing.""" raise exceptions.NotSupportedError()
class Playing
-
Base class for retrieving what is currently playing.
Expand source code
class Playing(ABC): """Base class for retrieving what is currently playing.""" def __str__(self) -> str: """Convert this playing object to a readable string.""" output = [] output.append( " Media type: {0}".format(convert.media_type_str(self.media_type)) ) output.append( "Device state: {0}".format(convert.device_state_str(self.device_state)) ) if self.title is not None: output.append(" Title: {0}".format(self.title)) if self.artist is not None: output.append(" Artist: {0}".format(self.artist)) if self.album is not None: output.append(" Album: {0}".format(self.album)) if self.genre is not None: output.append(" Genre: {0}".format(self.genre)) position = self.position total_time = self.total_time if position is not None and total_time is not None and total_time != 0: output.append( " Position: {0}/{1}s ({2:.1%})".format( position, total_time, float(position) / float(total_time) ) ) elif position is not None and position != 0: output.append(" Position: {0}s".format(position)) elif total_time is not None and position != 0: output.append(" Total time: {0}s".format(total_time)) if self.repeat is not None: output.append(" Repeat: {0}".format(convert.repeat_str(self.repeat))) if self.shuffle is not None: output.append(" Shuffle: {0}".format(convert.shuffle_str(self.shuffle))) return "\n".join(output) @property def hash(self) -> str: """Create a unique hash for what is currently playing. The hash is based on title, artist, album and total time. It should always be the same for the same content, but it is not guaranteed. """ base = "{0}{1}{2}{3}".format( self.title, self.artist, self.album, self.total_time ) return hashlib.sha256(base.encode("utf-8")).hexdigest() @property @abstractmethod def media_type(self) -> const.MediaType: """Type of media is currently playing, e.g. video, music.""" raise exceptions.NotSupportedError() @property @abstractmethod def device_state(self) -> const.DeviceState: """Device state, e.g. playing or paused.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(22, "Title", "Title of playing media.") def title(self) -> Optional[str]: """Title of the current media, e.g. movie or song name.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(23, "Artist", "Artist of playing song.") def artist(self) -> Optional[str]: """Artist of the currently playing song.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(24, "Album", "Album from playing artist.") def album(self) -> Optional[str]: """Album of the currently playing song.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(25, "Genre", "Genre of playing song.") def genre(self) -> Optional[str]: """Genre of the currently playing song.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(26, "TotalTime", "Total length of playing media (seconds).") def total_time(self) -> int: """Total play time in seconds.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(27, "Position", "Current play time position.") def position(self) -> int: """Position in the playing media (seconds).""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(28, "Shuffle", "Shuffle stat.e") def shuffle(self) -> const.ShuffleState: """If shuffle is enabled or not.""" raise exceptions.NotSupportedError() @property # type: ignore @abstractmethod @feature(29, "Repeat", "Repeat state.") def repeat(self) -> const.RepeatState: """Repeat mode.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapPlaying
- pyatv.mrp.MrpPlaying
Instance variables
var album
-
Album of the currently playing song.
Expand source code
@property # type: ignore @abstractmethod @feature(24, "Album", "Album from playing artist.") def album(self) -> Optional[str]: """Album of the currently playing song.""" raise exceptions.NotSupportedError()
var artist
-
Artist of the currently playing song.
Expand source code
@property # type: ignore @abstractmethod @feature(23, "Artist", "Artist of playing song.") def artist(self) -> Optional[str]: """Artist of the currently playing song.""" raise exceptions.NotSupportedError()
var device_state
-
Device state, e.g. playing or paused.
Expand source code
@property @abstractmethod def device_state(self) -> const.DeviceState: """Device state, e.g. playing or paused.""" raise exceptions.NotSupportedError()
var genre
-
Genre of the currently playing song.
Expand source code
@property # type: ignore @abstractmethod @feature(25, "Genre", "Genre of playing song.") def genre(self) -> Optional[str]: """Genre of the currently playing song.""" raise exceptions.NotSupportedError()
var hash
-
Create a unique hash for what is currently playing.
The hash is based on title, artist, album and total time. It should always be the same for the same content, but it is not guaranteed.
Expand source code
@property def hash(self) -> str: """Create a unique hash for what is currently playing. The hash is based on title, artist, album and total time. It should always be the same for the same content, but it is not guaranteed. """ base = "{0}{1}{2}{3}".format( self.title, self.artist, self.album, self.total_time ) return hashlib.sha256(base.encode("utf-8")).hexdigest()
var media_type
-
Type of media is currently playing, e.g. video, music.
Expand source code
@property @abstractmethod def media_type(self) -> const.MediaType: """Type of media is currently playing, e.g. video, music.""" raise exceptions.NotSupportedError()
var position
-
Position in the playing media (seconds).
Expand source code
@property # type: ignore @abstractmethod @feature(27, "Position", "Current play time position.") def position(self) -> int: """Position in the playing media (seconds).""" raise exceptions.NotSupportedError()
var repeat
-
Repeat mode.
Expand source code
@property # type: ignore @abstractmethod @feature(29, "Repeat", "Repeat state.") def repeat(self) -> const.RepeatState: """Repeat mode.""" raise exceptions.NotSupportedError()
var shuffle
-
If shuffle is enabled or not.
Expand source code
@property # type: ignore @abstractmethod @feature(28, "Shuffle", "Shuffle stat.e") def shuffle(self) -> const.ShuffleState: """If shuffle is enabled or not.""" raise exceptions.NotSupportedError()
var title
-
Title of the current media, e.g. movie or song name.
Expand source code
@property # type: ignore @abstractmethod @feature(22, "Title", "Title of playing media.") def title(self) -> Optional[str]: """Title of the current media, e.g. movie or song name.""" raise exceptions.NotSupportedError()
var total_time
-
Total play time in seconds.
Expand source code
@property # type: ignore @abstractmethod @feature(26, "TotalTime", "Total length of playing media (seconds).") def total_time(self) -> int: """Total play time in seconds.""" raise exceptions.NotSupportedError()
class Power
-
Base class for retrieving power state from an Apple TV.
Initialize a new Power instance.
Expand source code
class Power(ABC): """Base class for retrieving power state from an Apple TV.""" def __init__(self) -> None: """Initialize a new Power instance.""" self.__listener: Optional[PowerListener] = None @property def listener(self) -> Optional[PowerListener]: """Object receiving power state updates. Must be an object conforming to PowerListener. """ return self.__listener @listener.setter def listener(self, listener: Optional[PowerListener]) -> None: """Object that receives updates. This should be an object implementing method: - powerstate_update(old_state, new_state) """ self.__listener = listener @property # type: ignore @abstractmethod @feature(32, "PowerState", "Current device power state.") def power_state(self) -> const.PowerState: """Return device power state.""" raise exceptions.NotSupportedError() @abstractmethod @feature(33, "TurnOn", "Turn device on.") async def turn_on(self) -> None: """Turn device on.""" raise exceptions.NotSupportedError() @abstractmethod @feature(34, "TurnOff", "Turn off device.") async def turn_off(self) -> None: """Turn device off.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapPower
- pyatv.mrp.MrpPower
Instance variables
var listener
-
Object receiving power state updates.
Must be an object conforming to PowerListener.
Expand source code
@property def listener(self) -> Optional[PowerListener]: """Object receiving power state updates. Must be an object conforming to PowerListener. """ return self.__listener
var power_state
-
Return device power state.
Expand source code
@property # type: ignore @abstractmethod @feature(32, "PowerState", "Current device power state.") def power_state(self) -> const.PowerState: """Return device power state.""" raise exceptions.NotSupportedError()
Methods
async def turn_off(self)
-
Turn device off.
Expand source code
@abstractmethod @feature(34, "TurnOff", "Turn off device.") async def turn_off(self) -> None: """Turn device off.""" raise exceptions.NotSupportedError()
async def turn_on(self)
-
Turn device on.
Expand source code
@abstractmethod @feature(33, "TurnOn", "Turn device on.") async def turn_on(self) -> None: """Turn device on.""" raise exceptions.NotSupportedError()
class PowerListener
-
Listener interface for power updates.
Expand source code
class PowerListener(ABC): # pylint: disable=too-few-public-methods """Listener interface for power updates.""" @abstractmethod def powerstate_update( self, old_state: const.PowerState, new_state: const.PowerState ): """Device power state was updated.""" raise NotImplementedError()
Ancestors
- abc.ABC
Methods
def powerstate_update(self, old_state, new_state)
-
Device power state was updated.
Expand source code
@abstractmethod def powerstate_update( self, old_state: const.PowerState, new_state: const.PowerState ): """Device power state was updated.""" raise NotImplementedError()
class PushListener
-
Listener interface for push updates.
Expand source code
class PushListener(ABC): """Listener interface for push updates.""" @abstractmethod def playstatus_update(self, updater, playstatus: Playing) -> None: """Inform about changes to what is currently playing.""" @abstractmethod def playstatus_error(self, updater, exception: Exception) -> None: """Inform about an error when updating play status."""
Ancestors
- abc.ABC
Subclasses
- pyatv.scripts.atvremote.PushListener
- pyatv.scripts.atvscript.PushPrinter
Methods
def playstatus_error(self, updater, exception)
-
Inform about an error when updating play status.
Expand source code
@abstractmethod def playstatus_error(self, updater, exception: Exception) -> None: """Inform about an error when updating play status."""
def playstatus_update(self, updater, playstatus)
-
Inform about changes to what is currently playing.
Expand source code
@abstractmethod def playstatus_update(self, updater, playstatus: Playing) -> None: """Inform about changes to what is currently playing."""
class PushUpdater
-
Base class for push/async updates from an Apple TV.
Initialize a new PushUpdater.
Expand source code
class PushUpdater(ABC): """Base class for push/async updates from an Apple TV.""" def __init__(self) -> None: """Initialize a new PushUpdater.""" self.__listener: Optional[PushListener] = None @property @abstractmethod def active(self) -> bool: """Return if push updater has been started.""" raise exceptions.NotSupportedError() @property def listener(self) -> Optional[PushListener]: """Object (PushUpdaterListener) that receives updates.""" return self.__listener @listener.setter def listener(self, listener: Optional[PushListener]) -> None: """Object that receives updates. This should be an object implementing two methods: - playstatus_update(updater, playstatus) - playstatus_error(updater, exception) The first method is called when a new update happens and the second one is called if an error occurs. """ self.__listener = listener @abstractmethod def start(self, initial_delay: int = 0) -> None: """Begin to listen to updates. If an error occurs, start must be called again. """ raise exceptions.NotSupportedError() @abstractmethod def stop(self) -> None: """No longer forward updates to listener.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapPushUpdater
- pyatv.mrp.MrpPushUpdater
Instance variables
var active
-
Return if push updater has been started.
Expand source code
@property @abstractmethod def active(self) -> bool: """Return if push updater has been started.""" raise exceptions.NotSupportedError()
var listener
-
Object (PushUpdaterListener) that receives updates.
Expand source code
@property def listener(self) -> Optional[PushListener]: """Object (PushUpdaterListener) that receives updates.""" return self.__listener
Methods
def start(self, initial_delay=0)
-
Begin to listen to updates.
If an error occurs, start must be called again.
Expand source code
@abstractmethod def start(self, initial_delay: int = 0) -> None: """Begin to listen to updates. If an error occurs, start must be called again. """ raise exceptions.NotSupportedError()
def stop(self)
-
No longer forward updates to listener.
Expand source code
@abstractmethod def stop(self) -> None: """No longer forward updates to listener.""" raise exceptions.NotSupportedError()
class RemoteControl
-
Base class for API used to control an Apple TV.
Expand source code
class RemoteControl(ABC): # pylint: disable=too-many-public-methods """Base class for API used to control an Apple TV.""" # pylint: disable=invalid-name @abstractmethod @feature(0, "Up", "Up button on remote.") def up(self) -> None: """Press key up.""" raise exceptions.NotSupportedError() @abstractmethod @feature(1, "Down", "Down button on remote.") def down(self) -> None: """Press key down.""" raise exceptions.NotSupportedError() @abstractmethod @feature(2, "Left", "Left button on remote.") def left(self) -> None: """Press key left.""" raise exceptions.NotSupportedError() @abstractmethod @feature(3, "Right", "Right button on remote.") def right(self) -> None: """Press key right.""" raise exceptions.NotSupportedError() @abstractmethod @feature(4, "Play", "Start playing media.") def play(self) -> None: """Press key play.""" raise exceptions.NotSupportedError() @abstractmethod @feature(5, "PlayPause", "Toggle between play/pause.") def play_pause(self) -> None: """Toggle between play and pause.""" raise exceptions.NotSupportedError() @abstractmethod @feature(6, "Pause", "Pause playing media.") def pause(self) -> None: """Press key play.""" raise exceptions.NotSupportedError() @abstractmethod @feature(7, "Stop", "Stop playing media.") def stop(self) -> None: """Press key stop.""" raise exceptions.NotSupportedError() @abstractmethod @feature(8, "Next", "Change to next item.") def next(self) -> None: """Press key next.""" raise exceptions.NotSupportedError() @abstractmethod @feature(9, "Previous", "Change to previous item.") def previous(self) -> None: """Press key previous.""" raise exceptions.NotSupportedError() @abstractmethod @feature(10, "Select", "Select current option.") def select(self) -> None: """Press key select.""" raise exceptions.NotSupportedError() @abstractmethod @feature(11, "Menu", "Go back to previos menu.") def menu(self) -> None: """Press key menu.""" raise exceptions.NotSupportedError() @abstractmethod @feature(12, "VolumeUp", "Increase volume.") def volume_up(self) -> None: """Press key volume up.""" raise exceptions.NotSupportedError() @abstractmethod @feature(13, "VolumeDown", "Decrease volume.") def volume_down(self) -> None: """Press key volume down.""" raise exceptions.NotSupportedError() @abstractmethod @feature(14, "Home", "Home/TV button.") def home(self) -> None: """Press key home.""" raise exceptions.NotSupportedError() @abstractmethod @feature(15, "HomeHold", "Long-press home button.") def home_hold(self) -> None: """Hold key home.""" raise exceptions.NotSupportedError() @abstractmethod @feature(16, "TopMenu", "Go to main menu.") def top_menu(self) -> None: """Go to main menu (long press menu).""" raise exceptions.NotSupportedError() @abstractmethod @feature(17, "Suspend", "Suspend device (deprecated; use Power.turn_off).") def suspend(self) -> None: """Suspend the device.""" raise exceptions.NotSupportedError() @abstractmethod @feature(18, "WakeUp", "Wake up device (deprecated; use Power.turn_on).") def wakeup(self) -> None: """Wake up the device.""" raise exceptions.NotSupportedError() @abstractmethod @feature(19, "SetPosition", "Seek to position.") def set_position(self, pos) -> None: """Seek in the current playing media.""" raise exceptions.NotSupportedError() @abstractmethod @feature(20, "SetShuffle", "Change shuffle state.") def set_shuffle(self, shuffle_state) -> None: """Change shuffle mode to on or off.""" raise exceptions.NotSupportedError() @abstractmethod @feature(21, "SetRepeat", "Change repeat state.") def set_repeat(self, repeat_state) -> None: """Change repeat state.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.dmap.DmapRemoteControl
- pyatv.mrp.MrpRemoteControl
Methods
def down(self)
-
Press key down.
Expand source code
@abstractmethod @feature(1, "Down", "Down button on remote.") def down(self) -> None: """Press key down.""" raise exceptions.NotSupportedError()
def home(self)
-
Press key home.
Expand source code
@abstractmethod @feature(14, "Home", "Home/TV button.") def home(self) -> None: """Press key home.""" raise exceptions.NotSupportedError()
def home_hold(self)
-
Hold key home.
Expand source code
@abstractmethod @feature(15, "HomeHold", "Long-press home button.") def home_hold(self) -> None: """Hold key home.""" raise exceptions.NotSupportedError()
def left(self)
-
Press key left.
Expand source code
@abstractmethod @feature(2, "Left", "Left button on remote.") def left(self) -> None: """Press key left.""" raise exceptions.NotSupportedError()
-
Press key menu.
Expand source code
@abstractmethod @feature(11, "Menu", "Go back to previos menu.") def menu(self) -> None: """Press key menu.""" raise exceptions.NotSupportedError()
def next(self)
-
Press key next.
Expand source code
@abstractmethod @feature(8, "Next", "Change to next item.") def next(self) -> None: """Press key next.""" raise exceptions.NotSupportedError()
def pause(self)
-
Press key play.
Expand source code
@abstractmethod @feature(6, "Pause", "Pause playing media.") def pause(self) -> None: """Press key play.""" raise exceptions.NotSupportedError()
def play(self)
-
Press key play.
Expand source code
@abstractmethod @feature(4, "Play", "Start playing media.") def play(self) -> None: """Press key play.""" raise exceptions.NotSupportedError()
def play_pause(self)
-
Toggle between play and pause.
Expand source code
@abstractmethod @feature(5, "PlayPause", "Toggle between play/pause.") def play_pause(self) -> None: """Toggle between play and pause.""" raise exceptions.NotSupportedError()
def previous(self)
-
Press key previous.
Expand source code
@abstractmethod @feature(9, "Previous", "Change to previous item.") def previous(self) -> None: """Press key previous.""" raise exceptions.NotSupportedError()
def right(self)
-
Press key right.
Expand source code
@abstractmethod @feature(3, "Right", "Right button on remote.") def right(self) -> None: """Press key right.""" raise exceptions.NotSupportedError()
def select(self)
-
Press key select.
Expand source code
@abstractmethod @feature(10, "Select", "Select current option.") def select(self) -> None: """Press key select.""" raise exceptions.NotSupportedError()
def set_position(self, pos)
-
Seek in the current playing media.
Expand source code
@abstractmethod @feature(19, "SetPosition", "Seek to position.") def set_position(self, pos) -> None: """Seek in the current playing media.""" raise exceptions.NotSupportedError()
def set_repeat(self, repeat_state)
-
Change repeat state.
Expand source code
@abstractmethod @feature(21, "SetRepeat", "Change repeat state.") def set_repeat(self, repeat_state) -> None: """Change repeat state.""" raise exceptions.NotSupportedError()
def set_shuffle(self, shuffle_state)
-
Change shuffle mode to on or off.
Expand source code
@abstractmethod @feature(20, "SetShuffle", "Change shuffle state.") def set_shuffle(self, shuffle_state) -> None: """Change shuffle mode to on or off.""" raise exceptions.NotSupportedError()
def stop(self)
-
Press key stop.
Expand source code
@abstractmethod @feature(7, "Stop", "Stop playing media.") def stop(self) -> None: """Press key stop.""" raise exceptions.NotSupportedError()
def suspend(self)
-
Suspend the device.
Expand source code
@abstractmethod @feature(17, "Suspend", "Suspend device (deprecated; use Power.turn_off).") def suspend(self) -> None: """Suspend the device.""" raise exceptions.NotSupportedError()
-
Go to main menu (long press menu).
Expand source code
@abstractmethod @feature(16, "TopMenu", "Go to main menu.") def top_menu(self) -> None: """Go to main menu (long press menu).""" raise exceptions.NotSupportedError()
def up(self)
-
Press key up.
Expand source code
@abstractmethod @feature(0, "Up", "Up button on remote.") def up(self) -> None: """Press key up.""" raise exceptions.NotSupportedError()
def volume_down(self)
-
Press key volume down.
Expand source code
@abstractmethod @feature(13, "VolumeDown", "Decrease volume.") def volume_down(self) -> None: """Press key volume down.""" raise exceptions.NotSupportedError()
def volume_up(self)
-
Press key volume up.
Expand source code
@abstractmethod @feature(12, "VolumeUp", "Increase volume.") def volume_up(self) -> None: """Press key volume up.""" raise exceptions.NotSupportedError()
def wakeup(self)
-
Wake up the device.
Expand source code
@abstractmethod @feature(18, "WakeUp", "Wake up device (deprecated; use Power.turn_on).") def wakeup(self) -> None: """Wake up the device.""" raise exceptions.NotSupportedError()
class Stream
-
Base class for stream functionality.
Expand source code
class Stream(ABC): # pylint: disable=too-few-public-methods """Base class for stream functionality.""" @abstractmethod @feature(31, "PlayUrl", "Stream a URL on device.") def play_url(self, url: str, **kwargs) -> None: """Play media from an URL on the device.""" raise exceptions.NotSupportedError()
Ancestors
- abc.ABC
Subclasses
- pyatv.airplay.AirPlayStreamAPI
Methods
def play_url(self, url, **kwargs)
-
Play media from an URL on the device.
Expand source code
@abstractmethod @feature(31, "PlayUrl", "Stream a URL on device.") def play_url(self, url: str, **kwargs) -> None: """Play media from an URL on the device.""" raise exceptions.NotSupportedError()