platypush/platypush/plugins/media/plex/__init__.py

483 lines
No EOL
14 KiB
Python

import urllib.parse
from platypush.context import get_plugin
from platypush.plugins import Plugin, action
class MediaPlexPlugin(Plugin):
"""
Plugin to interact with a Plex media server
Requires:
* **plexapi** (``pip install plexapi``)
"""
def __init__(self, server, username, password, **kwargs):
"""
:param server: Plex server name
:type server: str
:param username: Plex username
:type username: str
:param password: Plex password
:type username: str
"""
from plexapi.myplex import MyPlexAccount
super().__init__(**kwargs)
self.resource = MyPlexAccount(username, password).resource(server)
self._plex = None
@property
def plex(self):
if not self._plex:
self._plex = self.resource.connect()
return self._plex
@action
def get_clients(self):
"""
Get the list of active clients
"""
return [{
'device': c.device,
'device_class': c.deviceClass,
'local': c.local,
'model': c.model,
'platform': c.platform,
'platform_version': c.platformVersion,
'product': c.product,
'state': c.state,
'title': c.title,
'version': c.version,
} for c in self.plex.clients()]
def _get_client(self, name):
return self.plex.client(name)
@action
def search(self, section=None, title=None, **kwargs):
"""
Return all the items matching the search criteria (default: all library items)
:param section: Section to search (Movies, Shows etc.)
:type section: str
:param title: Full or partial title
:type title: str
:param kwargs: Search criteria - includes e.g. title, unwatched, director, genre etc.
:type kwargs: dict
"""
ret = []
library = self.plex.library
if section:
library = library.section(section)
if title or kwargs:
items = library.search(title, **kwargs)
else:
items = library.all()
for item in items:
ret.append(self._flatten_item(item))
return ret
@action
def playlists(self):
"""
Get the playlists on the server
"""
return [
{
'title': pl.title,
'duration': pl.duration,
'summary': pl.summary,
'viewed_at': pl.viewedAt,
'items': [self._flatten_item(item) for item in pl.items()],
} for pl in self.plex.playlists()
]
@action
def history(self):
"""
Get the history of items played on the server
"""
return [
self._flatten_item(item) for item in self.plex.history()
]
@staticmethod
def get_chromecast(chromecast):
from ..lib.plexcast import PlexController
hndl = PlexController()
hndl.namespace = 'urn:x-cast:com.google.cast.sse'
cast = get_plugin('media.chromecast').get_chromecast(chromecast)
cast.register_handler(hndl)
return cast, hndl
@action
def play(self, client=None, chromecast=None, **kwargs):
"""
Search and play content on a client or a Chromecast. If no search filter
is specified, a play event will be sent to the specified client.
NOTE: Adding and managing play queues through the Plex API isn't fully
supported yet, therefore in case multiple items are returned from the
search only the first one will be played.
:param client: Client name
:type client: str
:param chromecast: Chromecast name
:type chromecast: str
:param kwargs: Search filter (e.g. title, section, unwatched, director etc.)
:type kwargs: dict
"""
if not client and not chromecast:
raise RuntimeError('No client nor chromecast specified')
if client:
client = self.plex.client(client)
elif chromecast:
(chromecast, handler) = self.get_chromecast(chromecast)
if not kwargs:
if client:
return client.play()
elif chromecast:
return handler.play()
if 'section' in kwargs:
library = self.plex.library.section(kwargs.pop('section'))
else:
library = self.plex.library
results = library.search(**kwargs)
if not results:
self.logger.info('No results for {}'.format(kwargs))
return
item = results[0]
self.logger.info('Playing {} on {}'.format(item.title, client or chromecast))
if client:
return client.playMedia(item)
elif chromecast:
return handler.play_media(item, self.plex)
@action
def pause(self, client):
"""
Send pause event to a client
"""
return self.client(client).pause()
@action
def stop(self, client):
"""
Send stop event to a client
"""
return self.client(client).stop()
@action
def seek(self, client, offset):
"""
Send seek event to a client
"""
return self.client(client).seekTo(offset)
@action
def forward(self, client):
"""
Forward playback on a client
"""
return self.client(client).stepForward()
@action
def back(self, client):
"""
Backward playback on a client
"""
return self.client(client).stepBack()
@action
def next(self, client):
"""
Play next item on a client
"""
return self.client(client).skipNext()
@action
def previous(self, client):
"""
Play previous item on a client
"""
return self.client(client).skipPrevious()
@action
def set_volume(self, client, volume):
"""
Set the volume on a client between 0 and 100
"""
return self.client(client).setVolume(volume / 100)
@action
def repeat(self, client, repeat):
"""
Set the repeat status on a client
"""
return self.client(client).setRepeat(repeat)
@action
def random(self, client, random):
"""
Set the random status on a client
"""
return self.client(client).setShuffle(random)
@action
def up(self, client):
"""
Send an up key event to a client
"""
return self.client(client).moveUp()
@action
def down(self, client):
"""
Send a down key event to a client
"""
return self.client(client).moveDown()
@action
def left(self, client):
"""
Send a left key event to a client
"""
return self.client(client).moveLeft()
@action
def right(self, client):
"""
Send a right key event to a client
"""
return self.client(client).moveRight()
@action
def go_back(self, client):
"""
Send a back key event to a client
"""
return self.client(client).goBack()
@action
def go_home(self, client):
"""
Send a home key event to a client
"""
return self.client(client).goHome()
@action
def go_to_media(self, client):
"""
Send a go to media event to a client
"""
return self.client(client).goToMedia()
@action
def go_to_music(self, client):
"""
Send a go to music event to a client
"""
return self.client(client).goToMusic()
@action
def next_letter(self, client):
"""
Send a next letter event to a client
"""
return self.client(client).nextLetter()
@action
def page_down(self, client):
"""
Send a page down event to a client
"""
return self.client(client).pageDown()
@action
def page_up(self, client):
"""
Send a page up event to a client
"""
return self.client(client).pageUp()
def _flatten_item(self, item):
from plexapi.audio import Track
from plexapi.video import Movie, Show
_item = {
'summary': item.summary,
'title': item.title,
'type': item.type,
'genres': [g.tag for g in getattr(item, 'genres', [])],
'art': getattr(item, 'art', None),
'art_url': getattr(item, 'artUrl', None),
'rating': getattr(item, 'rating', None),
'content_rating': getattr(item, 'content_rating', None),
}
if isinstance(item, Movie):
_item['is_watched'] = item.isWatched
_item['view_offset'] = item.viewOffset
_item['view_count'] = item.viewCount
_item['year'] = item.year
_item['audience_rating'] = item.audienceRating
_item['countries'] = [c.tag for c in item.countries]
_item['media'] = [
{
'duration': (item.media[i].duration or 0) / 1000,
'width': item.media[i].width,
'height': item.media[i].height,
'audio_channels': item.media[i].audioChannels,
'audio_codec': item.media[i].audioCodec,
'video_codec': item.media[i].videoCodec,
'video_resolution': item.media[i].videoResolution,
'video_frame_rate': item.media[i].videoFrameRate,
'parts': [
{
'file': part.file,
'size': part.size,
'duration': (part.duration or 0) / 1000,
'url': self.plex.url(part.key) + '?' + urllib.parse.urlencode({
'X-Plex-Token': self.plex._token,
}),
} for part in item.media[i].parts
]
} for i in range(0, len(item.media))
]
elif isinstance(item, Show):
_item['media'] = [
{
'title': season.title,
'season_number': season.seasonNumber,
'summary': season.summary,
'episodes': [
{
'duration': episode.duration / 1000,
'index': episode.index,
'year': episode.year,
'season_number': episode.seasonNumber,
'season_episode': episode.seasonEpisode,
'summary': episode.summary,
'is_watched': episode.isWatched,
'view_count': episode.viewCount,
'view_offset': episode.viewOffset,
'media': [
{
'duration': episode.media[i].duration / 1000,
'width': episode.media[i].width,
'height': episode.media[i].height,
'audio_channels': episode.media[i].audioChannels,
'audio_codec': episode.media[i].audioCodec,
'video_codec': episode.media[i].videoCodec,
'video_resolution': episode.media[i].videoResolution,
'video_frame_rate': episode.media[i].videoFrameRate,
'title': episode.title,
'parts': [
{
'file': part.file,
'size': part.size,
'duration': part.duration / 1000,
'url': self.plex.url(part.key) + '?' + urllib.parse.urlencode({
'X-Plex-Token': self.plex._token,
}),
} for part in episode.media[i].parts
]
} for i in range(0, len(episode.locations))
]
} for episode in season.episodes()
]
} for season in item.seasons()
]
elif isinstance(item, Track):
_item.update({
'artist': item.grandparentTitle,
'album': item.parentTitle,
'title': item.title,
'name': item.title,
'duration': item.duration / 1000.,
'index': item.index,
'track_number': item.trackNumber,
'year': item.year,
'locations': [item.locations],
})
_item['media'] = [
{
'title': media.title,
'duration': media.duration / 1000.,
'bitrate': media.bitrate,
'width': media.width,
'height': media.height,
'audio_channels': media.audioChannels,
'audio_codec': media.audioCodec,
'video_codec': media.videoCodec,
'video_resolution': media.videoResolution,
'video_frame_rate': media.videoFrameRate,
'parts': [
{
'file': part.file,
'duration': part.duration / 1000,
'size': part.size,
'url': self.plex.url(part.key) + '?' + urllib.parse.urlencode({
'X-Plex-Token': self.plex._token,
}),
} for part in media.parts
]
} for media in item.media
]
return _item
# vim:sw=4:ts=4:et: