python-plexapi/plexapi/server.py

268 lines
9.5 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2014-12-29 03:21:58 +00:00
"""
PlexServer
"""
2016-12-15 23:06:12 +00:00
from xml.etree import ElementTree
import requests
2014-12-29 03:21:58 +00:00
from requests.status_codes import _codes as codes
2016-12-15 23:06:12 +00:00
2014-12-29 03:21:58 +00:00
from plexapi import BASE_HEADERS, TIMEOUT
from plexapi import log, utils
2016-12-15 23:55:48 +00:00
from plexapi import audio, video, photo, playlist # noqa; required # why is this needed?
from plexapi.client import PlexClient
2016-12-15 23:06:12 +00:00
from plexapi.compat import quote
2014-12-29 03:21:58 +00:00
from plexapi.exceptions import BadRequest, NotFound
from plexapi.library import Library
2016-04-11 03:49:23 +00:00
from plexapi.playlist import Playlist
2014-12-29 03:21:58 +00:00
from plexapi.playqueue import PlayQueue
class PlexServer(object):
2016-12-15 23:55:48 +00:00
"""Main class to interact with plexapi.
Examples:
>>>> plex = PlexServer(token=12345)
>>>> for client in plex.clients():
>>>> print(client.title)
Note:
See test/example.py for more examples
Attributes:
baseurl (string): Base url for PMS
friendlyName (string): Pretty name for PMS
machineIdentifier (string): uuid for PMS
myPlex (TYPE): Description
myPlexMappingState (TYPE): Description
myPlexSigninState (TYPE): Description
myPlexSubscription (TYPE): Description
myPlexUsername (string): Description
platform (string): Description
platformVersion (string): Description
session (requests.Session, optinal): Add your own session object for caching
token (string): X-Plex-Token, using for authenication with PMS
transcoderActiveVideoSessions (int): How any active video sessions
updatedAt (int): Last updated at
version (TYPE): Description
2016-12-15 23:06:12 +00:00
"""
def __init__(self, baseurl='http://localhost:32400', token=None, session=None):
2016-12-15 23:55:48 +00:00
"""
Args:
baseurl (string): Base url for PMS
token (string): X-Plex-Token, using for authenication with PMS
session (requests.Session, optional): Use your own session object if you want
to cache the http responses from PMS
2016-12-15 23:06:12 +00:00
"""
self.baseurl = baseurl
2014-12-29 03:21:58 +00:00
self.token = token
2016-03-16 03:53:04 +00:00
self.session = session or requests.Session()
2014-12-29 03:21:58 +00:00
data = self._connect()
self.friendlyName = data.attrib.get('friendlyName')
self.machineIdentifier = data.attrib.get('machineIdentifier')
self.myPlex = bool(data.attrib.get('myPlex'))
self.myPlexMappingState = data.attrib.get('myPlexMappingState')
self.myPlexSigninState = data.attrib.get('myPlexSigninState')
self.myPlexSubscription = data.attrib.get('myPlexSubscription')
self.myPlexUsername = data.attrib.get('myPlexUsername')
self.platform = data.attrib.get('platform')
self.platformVersion = data.attrib.get('platformVersion')
self.transcoderActiveVideoSessions = int(data.attrib.get('transcoderActiveVideoSessions', 0))
self.updatedAt = int(data.attrib.get('updatedAt', 0))
2014-12-29 03:21:58 +00:00
self.version = data.attrib.get('version')
self._library = None # cached library
2014-12-29 03:21:58 +00:00
def __repr__(self):
2016-12-15 23:55:48 +00:00
"""Make repr prettier."""
return '<%s:%s>' % (self.__class__.__name__, self.baseurl)
2014-12-29 03:21:58 +00:00
def _connect(self):
2016-12-15 23:55:48 +00:00
"""Used for fetching the attributes for __init__."""
2014-12-29 03:21:58 +00:00
try:
return self.query('/')
2015-02-17 20:35:17 +00:00
except Exception as err:
log.error('%s: %s', self.baseurl, err)
raise NotFound('No server found at: %s' % self.baseurl)
2014-12-29 03:21:58 +00:00
@property
def library(self):
2016-12-15 23:55:48 +00:00
"""Library to browse or search your media."""
if not self._library:
self._library = Library(self, self.query('/library/'))
return self._library
2014-12-29 03:21:58 +00:00
def account(self):
data = self.query('/myplex/account')
return Account(self, data)
2014-12-29 03:21:58 +00:00
def clients(self):
2016-12-15 23:55:48 +00:00
"""Query PMS for all clients connected to PMS
2016-12-15 23:06:12 +00:00
2016-12-15 23:55:48 +00:00
Returns:
list: of Plexclient connnected to PMS
2016-12-15 23:06:12 +00:00
"""
2014-12-29 03:21:58 +00:00
items = []
for elem in self.query('/clients'):
2016-12-15 23:06:12 +00:00
baseurl = 'http://%s:%s' % (elem.attrib['address'],
elem.attrib['port'])
items.append(PlexClient(baseurl, server=self, data=elem))
2014-12-29 03:21:58 +00:00
return items
def client(self, name):
2016-12-15 23:06:12 +00:00
"""Querys PMS for all clients connected to PMS
2016-12-15 23:55:48 +00:00
Returns:
Plexclient
Args:
name (string): client title, John's Iphone
Raises:
NotFound: Unknown client name Betty
2016-12-15 23:06:12 +00:00
"""
2014-12-29 03:21:58 +00:00
for elem in self.query('/clients'):
if elem.attrib.get('name').lower() == name.lower():
2016-12-15 23:06:12 +00:00
baseurl = 'http://%s:%s' % (
elem.attrib['address'], elem.attrib['port'])
return PlexClient(baseurl, server=self, data=elem)
2014-12-29 03:21:58 +00:00
raise NotFound('Unknown client name: %s' % name)
2016-04-11 03:49:23 +00:00
def createPlaylist(self, title, items):
2016-12-15 23:55:48 +00:00
"""Create a playlist
Returns:
Playlist
"""
2016-04-11 03:49:23 +00:00
return Playlist.create(self, title, items)
def createPlayQueue(self, item):
return PlayQueue.create(self, item)
2014-12-29 03:21:58 +00:00
def headers(self):
2016-12-15 23:55:48 +00:00
"""Headers given to PMS."""
2014-12-29 03:21:58 +00:00
headers = BASE_HEADERS
if self.token:
headers['X-Plex-Token'] = self.token
return headers
2016-12-15 23:06:12 +00:00
def history(self):
2016-12-15 23:55:48 +00:00
"""List watched history.
"""
return utils.listItems(self, '/status/sessions/history/all')
2016-12-15 23:06:12 +00:00
def playlists(self):
2016-04-11 03:49:23 +00:00
# TODO: Add sort and type options?
# /playlists/all?type=15&sort=titleSort%3Aasc&playlistType=video&smart=0
return utils.listItems(self, '/playlists')
2016-12-15 23:06:12 +00:00
2016-12-15 23:55:48 +00:00
def playlist(self, title): # noqa
"""Returns a playlist with a given name or raise NotFound.
Args:
title (string): title of the playlist
Raises:
NotFound: Description
"""
for item in self.playlists():
if item.title == title:
return item
raise NotFound('Invalid playlist title: %s' % title)
2014-12-29 03:21:58 +00:00
def query(self, path, method=None, headers=None, **kwargs):
2016-12-15 23:55:48 +00:00
"""Main method used to handle http connection to PMS.
encodes the response to utf-8 and parses the xml returned
from PMS
Args:
path (sting): relative path to PMS, fx /search?query=HELLO
method (None, optional): requests.method, fx requests.put
headers (None, optional): Headers that will be passed to PMS
**kwargs (dict): Loads of different stuff
Raises:
BadRequest: Description
Returns:
ElementTree or None
"""
2014-12-29 03:21:58 +00:00
url = self.url(path)
2016-03-16 03:53:04 +00:00
method = method or self.session.get
2015-02-24 03:42:29 +00:00
log.info('%s %s', method.__name__.upper(), url)
2016-12-15 23:06:12 +00:00
h = headers.copy()
h.update(headers or {})
response = method(url, headers=h, timeout=TIMEOUT, **kwargs)
2014-12-29 03:21:58 +00:00
if response.status_code not in [200, 201]:
codename = codes.get(response.status_code)[0]
raise BadRequest('(%s) %s' % (response.status_code, codename))
data = response.text.encode('utf8')
return ElementTree.fromstring(data) if data else None
2016-12-15 23:06:12 +00:00
def search(self, query, mediatype=None):
2016-12-15 23:55:48 +00:00
"""Searching within a library section is much more powerful.
Args:
query (string): Search string
mediatype (string, optional): Limit your search to a media type.
Returns:
List
"""
items = utils.listItems(self, '/search?query=%s' % quote(query))
if mediatype:
return [item for item in items if item.type == mediatype]
return items
2015-02-17 20:35:17 +00:00
def sessions(self):
2016-12-15 23:55:48 +00:00
"""List all active sessions."""
return utils.listItems(self, '/status/sessions')
2015-02-17 20:35:17 +00:00
2014-12-29 03:21:58 +00:00
def url(self, path):
2016-12-15 23:55:48 +00:00
"""Build a full for PMS."""
2015-06-08 16:41:47 +00:00
if self.token:
delim = '&' if '?' in path else '?'
return '%s%s%sX-Plex-Token=%s' % (self.baseurl, path, delim, self.token)
return '%s%s' % (self.baseurl, path)
class Account(object):
2016-12-15 23:06:12 +00:00
"""This is the locally cached MyPlex account information. The properties provided don't match
2016-12-15 23:55:48 +00:00
the myplex.MyPlexAccount object very well. I believe this is here because access to myplex
is not required to get basic plex information.
Attributes:
authToken (TYPE): Description
mappingError (TYPE): Description
mappingErrorMessage (TYPE): Description
mappingState (TYPE): Description
privateAddress (TYPE): Description
privatePort (TYPE): Description
publicAddress (TYPE): Description
publicPort (TYPE): Description
signInState (TYPE): Description
subscriptionActive (TYPE): Description
subscriptionFeatures (TYPE): Description
subscriptionState (TYPE): Description
username (TYPE): Description
2016-12-15 23:06:12 +00:00
"""
def __init__(self, server, data):
self.authToken = data.attrib.get('authToken')
self.username = data.attrib.get('username')
self.mappingState = data.attrib.get('mappingState')
self.mappingError = data.attrib.get('mappingError')
self.mappingErrorMessage = data.attrib.get('mappingErrorMessage')
self.signInState = data.attrib.get('signInState')
self.publicAddress = data.attrib.get('publicAddress')
self.publicPort = data.attrib.get('publicPort')
self.privateAddress = data.attrib.get('privateAddress')
self.privatePort = data.attrib.get('privatePort')
self.subscriptionFeatures = data.attrib.get('subscriptionFeatures')
self.subscriptionActive = data.attrib.get('subscriptionActive')
self.subscriptionState = data.attrib.get('subscriptionState')