# -*- coding: utf-8 -*- import xml from urllib.parse import quote_plus from plexapi import log, settings, utils from plexapi.base import PlexObject from plexapi.exceptions import BadRequest from plexapi.utils import deprecated @utils.registerPlexObject class Media(PlexObject): """ Container object for all MediaPart objects. Provides useful data about the video or audio this media belong to such as video framerate, resolution, etc. Attributes: TAG (str): 'Media' aspectRatio (float): The aspect ratio of the media (ex: 2.35). audioChannels (int): The number of audio channels of the media (ex: 6). audioCodec (str): The audio codec of the media (ex: ac3). audioProfile (str): The audio profile of the media (ex: dts). bitrate (int): The bitrate of the media (ex: 1624). container (str): The container of the media (ex: avi). duration (int): The duration of the media in milliseconds (ex: 6990483). height (int): The height of the media in pixels (ex: 256). id (int): The unique ID for this media on the server. has64bitOffsets (bool): True if video has 64 bit offsets. optimizedForStreaming (bool): True if video is optimized for streaming. parts (List<:class:`~plexapi.media.MediaPart`>): List of media part objects. proxyType (int): Equals 42 for optimized versions. target (str): The media version target name. title (str): The title of the media. videoCodec (str): The video codec of the media (ex: ac3). videoFrameRate (str): The video frame rate of the media (ex: 24p). videoProfile (str): The video profile of the media (ex: high). videoResolution (str): The video resolution of the media (ex: sd). width (int): The width of the video in pixels (ex: 608). : The following attributes are only available for photos. * aperture (str): The aperture used to take the photo. * exposure (str): The exposure used to take the photo. * iso (int): The iso used to take the photo. * lens (str): The lens used to take the photo. * make (str): The make of the camera used to take the photo. * model (str): The model of the camera used to take the photo. """ TAG = 'Media' def _loadData(self, data): """ Load attribute values from Plex XML response. """ self._data = data self.aspectRatio = utils.cast(float, data.attrib.get('aspectRatio')) self.audioChannels = utils.cast(int, data.attrib.get('audioChannels')) self.audioCodec = data.attrib.get('audioCodec') self.audioProfile = data.attrib.get('audioProfile') self.bitrate = utils.cast(int, data.attrib.get('bitrate')) self.container = data.attrib.get('container') self.duration = utils.cast(int, data.attrib.get('duration')) self.height = utils.cast(int, data.attrib.get('height')) self.id = utils.cast(int, data.attrib.get('id')) self.has64bitOffsets = utils.cast(bool, data.attrib.get('has64bitOffsets')) self.optimizedForStreaming = utils.cast(bool, data.attrib.get('optimizedForStreaming')) self.parts = self.findItems(data, MediaPart) self.proxyType = utils.cast(int, data.attrib.get('proxyType')) self.target = data.attrib.get('target') self.title = data.attrib.get('title') self.videoCodec = data.attrib.get('videoCodec') self.videoFrameRate = data.attrib.get('videoFrameRate') self.videoProfile = data.attrib.get('videoProfile') self.videoResolution = data.attrib.get('videoResolution') self.width = utils.cast(int, data.attrib.get('width')) if self._isChildOf(etag='Photo'): self.aperture = data.attrib.get('aperture') self.exposure = data.attrib.get('exposure') self.iso = utils.cast(int, data.attrib.get('iso')) self.lens = data.attrib.get('lens') self.make = data.attrib.get('make') self.model = data.attrib.get('model') parent = self._parent() self._parentKey = parent.key @property def isOptimizedVersion(self): """ Returns True if the media is a Plex optimized version. """ return self.proxyType == utils.SEARCHTYPES['optimizedVersion'] def delete(self): part = '%s/media/%s' % (self._parentKey, self.id) try: return self._server.query(part, method=self._server._session.delete) except BadRequest: log.error("Failed to delete %s. This could be because you haven't allowed " "items to be deleted" % part) raise @utils.registerPlexObject class MediaPart(PlexObject): """ Represents a single media part (often a single file) for the media this belongs to. Attributes: TAG (str): 'Part' accessible (bool): True if the file is accessible. audioProfile (str): The audio profile of the file. container (str): The container type of the file (ex: avi). decision (str): Unknown. deepAnalysisVersion (int): The Plex deep analysis version for the file. duration (int): The duration of the file in milliseconds. exists (bool): True if the file exists. file (str): The path to this file on disk (ex: /media/Movies/Cars (2006)/Cars (2006).mkv) has64bitOffsets (bool): True if the file has 64 bit offsets. hasThumbnail (bool): True if the file (track) has an embedded thumbnail. id (int): The unique ID for this media part on the server. indexes (str, None): sd if the file has generated preview (BIF) thumbnails. key (str): API URL (ex: /library/parts/46618/1389985872/file.mkv). optimizedForStreaming (bool): True if the file is optimized for streaming. packetLength (int): The packet length of the file. requiredBandwidths (str): The required bandwidths to stream the file. size (int): The size of the file in bytes (ex: 733884416). streams (List<:class:`~plexapi.media.MediaPartStream`>): List of stream objects. syncItemId (int): The unique ID for this media part if it is synced. syncState (str): The sync state for this media part. videoProfile (str): The video profile of the file. """ TAG = 'Part' def _loadData(self, data): """ Load attribute values from Plex XML response. """ self._data = data self.accessible = utils.cast(bool, data.attrib.get('accessible')) self.audioProfile = data.attrib.get('audioProfile') self.container = data.attrib.get('container') self.decision = data.attrib.get('decision') self.deepAnalysisVersion = utils.cast(int, data.attrib.get('deepAnalysisVersion')) self.duration = utils.cast(int, data.attrib.get('duration')) self.exists = utils.cast(bool, data.attrib.get('exists')) self.file = data.attrib.get('file') self.has64bitOffsets = utils.cast(bool, data.attrib.get('has64bitOffsets')) self.hasThumbnail = utils.cast(bool, data.attrib.get('hasThumbnail')) self.id = utils.cast(int, data.attrib.get('id')) self.indexes = data.attrib.get('indexes') self.key = data.attrib.get('key') self.optimizedForStreaming = utils.cast(bool, data.attrib.get('optimizedForStreaming')) self.packetLength = utils.cast(int, data.attrib.get('packetLength')) self.requiredBandwidths = data.attrib.get('requiredBandwidths') self.size = utils.cast(int, data.attrib.get('size')) self.streams = self._buildStreams(data) self.syncItemId = utils.cast(int, data.attrib.get('syncItemId')) self.syncState = data.attrib.get('syncState') self.videoProfile = data.attrib.get('videoProfile') def _buildStreams(self, data): streams = [] for cls in (VideoStream, AudioStream, SubtitleStream, LyricStream): items = self.findItems(data, cls, streamType=cls.STREAMTYPE) streams.extend(items) return streams @property def hasPreviewThumbnails(self): """ Returns True if the media part has generated preview (BIF) thumbnails. """ return self.indexes == 'sd' def videoStreams(self): """ Returns a list of :class:`~plexapi.media.VideoStream` objects in this MediaPart. """ return [stream for stream in self.streams if isinstance(stream, VideoStream)] def audioStreams(self): """ Returns a list of :class:`~plexapi.media.AudioStream` objects in this MediaPart. """ return [stream for stream in self.streams if isinstance(stream, AudioStream)] def subtitleStreams(self): """ Returns a list of :class:`~plexapi.media.SubtitleStream` objects in this MediaPart. """ return [stream for stream in self.streams if isinstance(stream, SubtitleStream)] def lyricStreams(self): """ Returns a list of :class:`~plexapi.media.LyricStream` objects in this MediaPart. """ return [stream for stream in self.streams if isinstance(stream, LyricStream)] def setDefaultAudioStream(self, stream): """ Set the default :class:`~plexapi.media.AudioStream` for this MediaPart. Parameters: stream (:class:`~plexapi.media.AudioStream`): AudioStream to set as default """ if isinstance(stream, AudioStream): key = "/library/parts/%d?audioStreamID=%d&allParts=1" % (self.id, stream.id) else: key = "/library/parts/%d?audioStreamID=%d&allParts=1" % (self.id, stream) self._server.query(key, method=self._server._session.put) def setDefaultSubtitleStream(self, stream): """ Set the default :class:`~plexapi.media.SubtitleStream` for this MediaPart. Parameters: stream (:class:`~plexapi.media.SubtitleStream`): SubtitleStream to set as default. """ if isinstance(stream, SubtitleStream): key = "/library/parts/%d?subtitleStreamID=%d&allParts=1" % (self.id, stream.id) else: key = "/library/parts/%d?subtitleStreamID=%d&allParts=1" % (self.id, stream) self._server.query(key, method=self._server._session.put) def resetDefaultSubtitleStream(self): """ Set default subtitle of this MediaPart to 'none'. """ key = "/library/parts/%d?subtitleStreamID=0&allParts=1" % (self.id) self._server.query(key, method=self._server._session.put) class MediaPartStream(PlexObject): """ Base class for media streams. These consist of video, audio, subtitles, and lyrics. Attributes: bitrate (int): The bitrate of the stream. codec (str): The codec of the stream (ex: srt, ac3, mpeg4). default (bool): True if this is the default stream. displayTitle (str): The display title of the stream. extendedDisplayTitle (str): The extended display title of the stream. key (str): API URL (/library/streams/) id (int): The unique ID for this stream on the server. index (int): The index of the stream. language (str): The language of the stream (ex: English, ไทย). languageCode (str): The ASCII language code of the stream (ex: eng, tha). requiredBandwidths (str): The required bandwidths to stream the file. selected (bool): True if this stream is selected. streamType (int): The stream type (1= :class:`~plexapi.media.VideoStream`, 2= :class:`~plexapi.media.AudioStream`, 3= :class:`~plexapi.media.SubtitleStream`). title (str): The title of the stream. type (int): Alias for streamType. """ def _loadData(self, data): """ Load attribute values from Plex XML response. """ self._data = data self.bitrate = utils.cast(int, data.attrib.get('bitrate')) self.codec = data.attrib.get('codec') self.default = utils.cast(bool, data.attrib.get('default')) self.displayTitle = data.attrib.get('displayTitle') self.extendedDisplayTitle = data.attrib.get('extendedDisplayTitle') self.key = data.attrib.get('key') self.id = utils.cast(int, data.attrib.get('id')) self.index = utils.cast(int, data.attrib.get('index', '-1')) self.language = data.attrib.get('language') self.languageCode = data.attrib.get('languageCode') self.requiredBandwidths = data.attrib.get('requiredBandwidths') self.selected = utils.cast(bool, data.attrib.get('selected', '0')) self.streamType = utils.cast(int, data.attrib.get('streamType')) self.title = data.attrib.get('title') self.type = utils.cast(int, data.attrib.get('streamType')) @utils.registerPlexObject class VideoStream(MediaPartStream): """ Represents a video stream within a :class:`~plexapi.media.MediaPart`. Attributes: TAG (str): 'Stream' STREAMTYPE (int): 1 anamorphic (str): If the video is anamorphic. bitDepth (int): The bit depth of the video stream (ex: 8). cabac (int): The context-adaptive binary arithmetic coding. chromaLocation (str): The chroma location of the video stream. chromaSubsampling (str): The chroma subsampling of the video stream (ex: 4:2:0). codecID (str): The codec ID (ex: XVID). codedHeight (int): The coded height of the video stream in pixels. codedWidth (int): The coded width of the video stream in pixels. colorPrimaries (str): The color primaries of the video stream. colorRange (str): The color range of the video stream. colorSpace (str): The color space of the video stream (ex: bt2020). colorTrc (str): The color trc of the video stream. DOVIBLCompatID (int): Dolby Vision base layer compatibility ID. DOVIBLPresent (bool): True if Dolby Vision base layer is present. DOVIELPresent (bool): True if Dolby Vision enhancement layer is present. DOVILevel (int): Dolby Vision level. DOVIPresent (bool): True if Dolby Vision is present. DOVIProfile (int): Dolby Vision profile. DOVIRPUPresent (bool): True if Dolby Vision reference processing unit is present. DOVIVersion (float): The Dolby Vision version. duration (int): The duration of video stream in milliseconds. frameRate (float): The frame rate of the video stream (ex: 23.976). frameRateMode (str): The frame rate mode of the video stream. hasScalingMatrix (bool): True if video stream has a scaling matrix. height (int): The height of the video stream in pixels (ex: 1080). level (int): The codec encoding level of the video stream (ex: 41). profile (str): The profile of the video stream (ex: asp). pixelAspectRatio (str): The pixel aspect ratio of the video stream. pixelFormat (str): The pixel format of the video stream. refFrames (int): The number of reference frames of the video stream. scanType (str): The scan type of the video stream (ex: progressive). streamIdentifier(int): The stream identifier of the video stream. width (int): The width of the video stream in pixels (ex: 1920). """ TAG = 'Stream' STREAMTYPE = 1 def _loadData(self, data): """ Load attribute values from Plex XML response. """ super(VideoStream, self)._loadData(data) self.anamorphic = data.attrib.get('anamorphic') self.bitDepth = utils.cast(int, data.attrib.get('bitDepth')) self.cabac = utils.cast(int, data.attrib.get('cabac')) self.chromaLocation = data.attrib.get('chromaLocation') self.chromaSubsampling = data.attrib.get('chromaSubsampling') self.codecID = data.attrib.get('codecID') self.codedHeight = utils.cast(int, data.attrib.get('codedHeight')) self.codedWidth = utils.cast(int, data.attrib.get('codedWidth')) self.colorPrimaries = data.attrib.get('colorPrimaries') self.colorRange = data.attrib.get('colorRange') self.colorSpace = data.attrib.get('colorSpace') self.colorTrc = data.attrib.get('colorTrc') self.DOVIBLCompatID = utils.cast(int, data.attrib.get('DOVIBLCompatID')) self.DOVIBLPresent = utils.cast(bool, data.attrib.get('DOVIBLPresent')) self.DOVIELPresent = utils.cast(bool, data.attrib.get('DOVIELPresent')) self.DOVILevel = utils.cast(int, data.attrib.get('DOVILevel')) self.DOVIPresent = utils.cast(bool, data.attrib.get('DOVIPresent')) self.DOVIProfile = utils.cast(int, data.attrib.get('DOVIProfile')) self.DOVIRPUPresent = utils.cast(bool, data.attrib.get('DOVIRPUPresent')) self.DOVIVersion = utils.cast(float, data.attrib.get('DOVIVersion')) self.duration = utils.cast(int, data.attrib.get('duration')) self.frameRate = utils.cast(float, data.attrib.get('frameRate')) self.frameRateMode = data.attrib.get('frameRateMode') self.hasScalingMatrix = utils.cast(bool, data.attrib.get('hasScalingMatrix')) self.height = utils.cast(int, data.attrib.get('height')) self.level = utils.cast(int, data.attrib.get('level')) self.profile = data.attrib.get('profile') self.pixelAspectRatio = data.attrib.get('pixelAspectRatio') self.pixelFormat = data.attrib.get('pixelFormat') self.refFrames = utils.cast(int, data.attrib.get('refFrames')) self.scanType = data.attrib.get('scanType') self.streamIdentifier = utils.cast(int, data.attrib.get('streamIdentifier')) self.width = utils.cast(int, data.attrib.get('width')) @utils.registerPlexObject class AudioStream(MediaPartStream): """ Represents a audio stream within a :class:`~plexapi.media.MediaPart`. Attributes: TAG (str): 'Stream' STREAMTYPE (int): 2 audioChannelLayout (str): The audio channel layout of the audio stream (ex: 5.1(side)). bitDepth (int): The bit depth of the audio stream (ex: 16). bitrateMode (str): The bitrate mode of the audio stream (ex: cbr). channels (int): The number of audio channels of the audio stream (ex: 6). duration (int): The duration of audio stream in milliseconds. profile (str): The profile of the audio stream. samplingRate (int): The sampling rate of the audio stream (ex: xxx) streamIdentifier (int): The stream identifier of the audio stream. : The following attributes are only available for tracks. * albumGain (float): The gain for the album. * albumPeak (float): The peak for the album. * albumRange (float): The range for the album. * endRamp (str): The end ramp for the track. * gain (float): The gain for the track. * loudness (float): The loudness for the track. * lra (float): The lra for the track. * peak (float): The peak for the track. * startRamp (str): The start ramp for the track. """ TAG = 'Stream' STREAMTYPE = 2 def _loadData(self, data): """ Load attribute values from Plex XML response. """ super(AudioStream, self)._loadData(data) self.audioChannelLayout = data.attrib.get('audioChannelLayout') self.bitDepth = utils.cast(int, data.attrib.get('bitDepth')) self.bitrateMode = data.attrib.get('bitrateMode') self.channels = utils.cast(int, data.attrib.get('channels')) self.duration = utils.cast(int, data.attrib.get('duration')) self.profile = data.attrib.get('profile') self.samplingRate = utils.cast(int, data.attrib.get('samplingRate')) self.streamIdentifier = utils.cast(int, data.attrib.get('streamIdentifier')) if self._isChildOf(etag='Track'): self.albumGain = utils.cast(float, data.attrib.get('albumGain')) self.albumPeak = utils.cast(float, data.attrib.get('albumPeak')) self.albumRange = utils.cast(float, data.attrib.get('albumRange')) self.endRamp = data.attrib.get('endRamp') self.gain = utils.cast(float, data.attrib.get('gain')) self.loudness = utils.cast(float, data.attrib.get('loudness')) self.lra = utils.cast(float, data.attrib.get('lra')) self.peak = utils.cast(float, data.attrib.get('peak')) self.startRamp = data.attrib.get('startRamp') @utils.registerPlexObject class SubtitleStream(MediaPartStream): """ Represents a audio stream within a :class:`~plexapi.media.MediaPart`. Attributes: TAG (str): 'Stream' STREAMTYPE (int): 3 container (str): The container of the subtitle stream. forced (bool): True if this is a forced subtitle. format (str): The format of the subtitle stream (ex: srt). headerCompression (str): The header compression of the subtitle stream. transient (str): Unknown. """ TAG = 'Stream' STREAMTYPE = 3 def _loadData(self, data): """ Load attribute values from Plex XML response. """ super(SubtitleStream, self)._loadData(data) self.container = data.attrib.get('container') self.forced = utils.cast(bool, data.attrib.get('forced', '0')) self.format = data.attrib.get('format') self.headerCompression = data.attrib.get('headerCompression') self.transient = data.attrib.get('transient') class LyricStream(MediaPartStream): """ Represents a lyric stream within a :class:`~plexapi.media.MediaPart`. Attributes: TAG (str): 'Stream' STREAMTYPE (int): 4 format (str): The format of the lyric stream (ex: lrc). minLines (int): The minimum number of lines in the (timed) lyric stream. provider (str): The provider of the lyric stream (ex: com.plexapp.agents.lyricfind). timed (bool): True if the lyrics are timed to the track. """ TAG = 'Stream' STREAMTYPE = 4 def _loadData(self, data): """ Load attribute values from Plex XML response. """ super(LyricStream, self)._loadData(data) self.format = data.attrib.get('format') self.minLines = utils.cast(int, data.attrib.get('minLines')) self.provider = data.attrib.get('provider') self.timed = utils.cast(bool, data.attrib.get('timed', '0')) @utils.registerPlexObject class Session(PlexObject): """ Represents a current session. Attributes: TAG (str): 'Session' id (str): The unique identifier for the session. bandwidth (int): The Plex streaming brain reserved bandwidth for the session. location (str): The location of the session (lan, wan, or cellular) """ TAG = 'Session' def _loadData(self, data): self.id = data.attrib.get('id') self.bandwidth = utils.cast(int, data.attrib.get('bandwidth')) self.location = data.attrib.get('location') @utils.registerPlexObject class TranscodeSession(PlexObject): """ Represents a current transcode session. Attributes: TAG (str): 'TranscodeSession' audioChannels (int): The number of audio channels of the transcoded media. audioCodec (str): The audio codec of the transcoded media. audioDecision (str): The transcode decision for the audio stream. complete (bool): True if the transcode is complete. container (str): The container of the transcoded media. context (str): The context for the transcode session. duration (int): The duration of the transcoded media in milliseconds. height (int): The height of the transcoded media in pixels. key (str): API URL (ex: /transcode/sessions/). maxOffsetAvailable (float): Unknown. minOffsetAvailable (float): Unknown. progress (float): The progress percentage of the transcode. protocol (str): The protocol of the transcode. remaining (int): Unknown. size (int): The size of the transcoded media in bytes. sourceAudioCodec (str): The audio codec of the source media. sourceVideoCodec (str): The video codec of the source media. speed (float): The speed of the transcode. subtitleDecision (str): The transcode decision for the subtitle stream throttled (bool): True if the transcode is throttled. timestamp (int): The epoch timestamp when the transcode started. transcodeHwDecoding (str): The hardware transcoding decoder engine. transcodeHwDecodingTitle (str): The title of the hardware transcoding decoder engine. transcodeHwEncoding (str): The hardware transcoding encoder engine. transcodeHwEncodingTitle (str): The title of the hardware transcoding encoder engine. transcodeHwFullPipeline (str): True if hardware decoding and encoding is being used for the transcode. transcodeHwRequested (str): True if hardware transcoding was requested for the transcode. videoCodec (str): The video codec of the transcoded media. videoDecision (str): The transcode decision for the video stream. width (str): The width of the transcoded media in pixels. """ TAG = 'TranscodeSession' def _loadData(self, data): """ Load attribute values from Plex XML response. """ self._data = data self.audioChannels = utils.cast(int, data.attrib.get('audioChannels')) self.audioCodec = data.attrib.get('audioCodec') self.audioDecision = data.attrib.get('audioDecision') self.complete = utils.cast(bool, data.attrib.get('complete', '0')) self.container = data.attrib.get('container') self.context = data.attrib.get('context') self.duration = utils.cast(int, data.attrib.get('duration')) self.height = utils.cast(int, data.attrib.get('height')) self.key = data.attrib.get('key') self.maxOffsetAvailable = utils.cast(float, data.attrib.get('maxOffsetAvailable')) self.minOffsetAvailable = utils.cast(float, data.attrib.get('minOffsetAvailable')) self.progress = utils.cast(float, data.attrib.get('progress')) self.protocol = data.attrib.get('protocol') self.remaining = utils.cast(int, data.attrib.get('remaining')) self.size = utils.cast(int, data.attrib.get('size')) self.sourceAudioCodec = data.attrib.get('sourceAudioCodec') self.sourceVideoCodec = data.attrib.get('sourceVideoCodec') self.speed = utils.cast(float, data.attrib.get('speed')) self.subtitleDecision = data.attrib.get('subtitleDecision') self.throttled = utils.cast(bool, data.attrib.get('throttled', '0')) self.timestamp = utils.cast(float, data.attrib.get('timeStamp')) self.transcodeHwDecoding = data.attrib.get('transcodeHwDecoding') self.transcodeHwDecodingTitle = data.attrib.get('transcodeHwDecodingTitle') self.transcodeHwEncoding = data.attrib.get('transcodeHwEncoding') self.transcodeHwEncodingTitle = data.attrib.get('transcodeHwEncodingTitle') self.transcodeHwFullPipeline = utils.cast(bool, data.attrib.get('transcodeHwFullPipeline', '0')) self.transcodeHwRequested = utils.cast(bool, data.attrib.get('transcodeHwRequested', '0')) self.videoCodec = data.attrib.get('videoCodec') self.videoDecision = data.attrib.get('videoDecision') self.width = utils.cast(int, data.attrib.get('width')) @utils.registerPlexObject class TranscodeJob(PlexObject): """ Represents an Optimizing job. TrancodeJobs are the process for optimizing conversions. Active or paused optimization items. Usually one item as a time.""" TAG = 'TranscodeJob' def _loadData(self, data): self._data = data self.generatorID = data.attrib.get('generatorID') self.key = data.attrib.get('key') self.progress = data.attrib.get('progress') self.ratingKey = data.attrib.get('ratingKey') self.size = data.attrib.get('size') self.targetTagID = data.attrib.get('targetTagID') self.thumb = data.attrib.get('thumb') self.title = data.attrib.get('title') self.type = data.attrib.get('type') @utils.registerPlexObject class Optimized(PlexObject): """ Represents a Optimized item. Optimized items are optimized and queued conversions items.""" TAG = 'Item' def _loadData(self, data): self._data = data self.id = data.attrib.get('id') self.composite = data.attrib.get('composite') self.title = data.attrib.get('title') self.type = data.attrib.get('type') self.target = data.attrib.get('target') self.targetTagID = data.attrib.get('targetTagID') def items(self): """ Returns a list of all :class:`~plexapi.media.Video` objects in this optimized item. """ key = '%s/%s/items' % (self._initpath, self.id) return self.fetchItems(key) def remove(self): """ Remove an Optimized item""" key = '%s/%s' % (self._initpath, self.id) self._server.query(key, method=self._server._session.delete) def rename(self, title): """ Rename an Optimized item""" key = '%s/%s?Item[title]=%s' % (self._initpath, self.id, title) self._server.query(key, method=self._server._session.put) def reprocess(self, ratingKey): """ Reprocess a removed Conversion item that is still a listed Optimize item""" key = '%s/%s/%s/enable' % (self._initpath, self.id, ratingKey) self._server.query(key, method=self._server._session.put) @utils.registerPlexObject class Conversion(PlexObject): """ Represents a Conversion item. Conversions are items queued for optimization or being actively optimized.""" TAG = 'Video' def _loadData(self, data): self._data = data self.addedAt = data.attrib.get('addedAt') self.art = data.attrib.get('art') self.chapterSource = data.attrib.get('chapterSource') self.contentRating = data.attrib.get('contentRating') self.duration = data.attrib.get('duration') self.generatorID = data.attrib.get('generatorID') self.generatorType = data.attrib.get('generatorType') self.guid = data.attrib.get('guid') self.key = data.attrib.get('key') self.lastViewedAt = data.attrib.get('lastViewedAt') self.librarySectionID = data.attrib.get('librarySectionID') self.librarySectionKey = data.attrib.get('librarySectionKey') self.librarySectionTitle = data.attrib.get('librarySectionTitle') self.originallyAvailableAt = data.attrib.get('originallyAvailableAt') self.playQueueItemID = data.attrib.get('playQueueItemID') self.playlistID = data.attrib.get('playlistID') self.primaryExtraKey = data.attrib.get('primaryExtraKey') self.rating = data.attrib.get('rating') self.ratingKey = data.attrib.get('ratingKey') self.studio = data.attrib.get('studio') self.summary = data.attrib.get('summary') self.tagline = data.attrib.get('tagline') self.target = data.attrib.get('target') self.thumb = data.attrib.get('thumb') self.title = data.attrib.get('title') self.type = data.attrib.get('type') self.updatedAt = data.attrib.get('updatedAt') self.userID = data.attrib.get('userID') self.username = data.attrib.get('username') self.viewOffset = data.attrib.get('viewOffset') self.year = data.attrib.get('year') def remove(self): """ Remove Conversion from queue """ key = '/playlists/%s/items/%s/%s/disable' % (self.playlistID, self.generatorID, self.ratingKey) self._server.query(key, method=self._server._session.put) def move(self, after): """ Move Conversion items position in queue after (int): Place item after specified playQueueItemID. '-1' is the active conversion. Example: Move 5th conversion Item to active conversion conversions[4].move('-1') Move 4th conversion Item to 3rd in conversion queue conversions[3].move(conversions[1].playQueueItemID) """ key = '%s/items/%s/move?after=%s' % (self._initpath, self.playQueueItemID, after) self._server.query(key, method=self._server._session.put) class MediaTag(PlexObject): """ Base class for media tags used for filtering and searching your library items or navigating the metadata of media items in your library. Tags are the construct used for things such as Country, Director, Genre, etc. Attributes: filter (str): The library filter for the tag. id (id): Tag ID (This seems meaningless except to use it as a unique id). key (str): API URL (/library/section//all?). role (str): The name of the character role for :class:`~plexapi.media.Role` only. tag (str): Name of the tag. This will be Animation, SciFi etc for Genres. The name of person for Directors and Roles (ex: Animation, Stephen Graham, etc). thumb (str): URL to thumbnail image for :class:`~plexapi.media.Role` only. """ def _loadData(self, data): """ Load attribute values from Plex XML response. """ self._data = data self.filter = data.attrib.get('filter') self.id = utils.cast(int, data.attrib.get('id')) self.key = data.attrib.get('key') self.role = data.attrib.get('role') self.tag = data.attrib.get('tag') self.thumb = data.attrib.get('thumb') parent = self._parent() self._librarySectionID = utils.cast(int, parent._data.attrib.get('librarySectionID')) self._librarySectionKey = parent._data.attrib.get('librarySectionKey') self._librarySectionTitle = parent._data.attrib.get('librarySectionTitle') self._parentType = parent.TYPE if self._librarySectionKey and self.filter: self.key = '%s/all?%s&type=%s' % ( self._librarySectionKey, self.filter, utils.searchType(self._parentType)) def items(self): """ Return the list of items within this tag. """ if not self.key: raise BadRequest('Key is not defined for this tag: %s. ' 'Reload the parent object.' % self.tag) return self.fetchItems(self.key) @utils.registerPlexObject class Collection(MediaTag): """ Represents a single Collection media tag. Attributes: TAG (str): 'Collection' FILTER (str): 'collection' """ TAG = 'Collection' FILTER = 'collection' def collection(self): """ Return the :class:`~plexapi.collection.Collection` object for this collection tag. """ key = '%s/collections' % self._librarySectionKey return self.fetchItem(key, etag='Directory', index=self.id) @utils.registerPlexObject class Country(MediaTag): """ Represents a single Country media tag. Attributes: TAG (str): 'Country' FILTER (str): 'country' """ TAG = 'Country' FILTER = 'country' @utils.registerPlexObject class Director(MediaTag): """ Represents a single Director media tag. Attributes: TAG (str): 'Director' FILTER (str): 'director' """ TAG = 'Director' FILTER = 'director' @utils.registerPlexObject class Format(MediaTag): """ Represents a single Format media tag. Attributes: TAG (str): 'Format' FILTER (str): 'format' """ TAG = 'Format' FILTER = 'format' @utils.registerPlexObject class Genre(MediaTag): """ Represents a single Genre media tag. Attributes: TAG (str): 'Genre' FILTER (str): 'genre' """ TAG = 'Genre' FILTER = 'genre' @utils.registerPlexObject class Label(MediaTag): """ Represents a single Label media tag. Attributes: TAG (str): 'Label' FILTER (str): 'label' """ TAG = 'Label' FILTER = 'label' @utils.registerPlexObject class Mood(MediaTag): """ Represents a single Mood media tag. Attributes: TAG (str): 'Mood' FILTER (str): 'mood' """ TAG = 'Mood' FILTER = 'mood' @utils.registerPlexObject class Producer(MediaTag): """ Represents a single Producer media tag. Attributes: TAG (str): 'Producer' FILTER (str): 'producer' """ TAG = 'Producer' FILTER = 'producer' @utils.registerPlexObject class Role(MediaTag): """ Represents a single Role (actor/actress) media tag. Attributes: TAG (str): 'Role' FILTER (str): 'role' """ TAG = 'Role' FILTER = 'role' @utils.registerPlexObject class Similar(MediaTag): """ Represents a single Similar media tag. Attributes: TAG (str): 'Similar' FILTER (str): 'similar' """ TAG = 'Similar' FILTER = 'similar' @utils.registerPlexObject class Style(MediaTag): """ Represents a single Style media tag. Attributes: TAG (str): 'Style' FILTER (str): 'style' """ TAG = 'Style' FILTER = 'style' @utils.registerPlexObject class Subformat(MediaTag): """ Represents a single Subformat media tag. Attributes: TAG (str): 'Subformat' FILTER (str): 'subformat' """ TAG = 'Subformat' FILTER = 'subformat' @utils.registerPlexObject class Tag(MediaTag): """ Represents a single Tag media tag. Attributes: TAG (str): 'Tag' FILTER (str): 'tag' """ TAG = 'Tag' FILTER = 'tag' @utils.registerPlexObject class Writer(MediaTag): """ Represents a single Writer media tag. Attributes: TAG (str): 'Writer' FILTER (str): 'writer' """ TAG = 'Writer' FILTER = 'writer' class GuidTag(PlexObject): """ Base class for guid tags used only for Guids, as they contain only a string identifier Attributes: id (id): The guid for external metadata sources (e.g. IMDB, TMDB, TVDB). """ def _loadData(self, data): """ Load attribute values from Plex XML response. """ self._data = data self.id = data.attrib.get('id') @utils.registerPlexObject class Guid(GuidTag): """ Represents a single Guid media tag. Attributes: TAG (str): 'Guid' """ TAG = 'Guid' @utils.registerPlexObject class Review(PlexObject): """ Represents a single Review for a Movie. Attributes: TAG (str): 'Review' filter (str): filter for reviews? id (int): The ID of the review. image (str): The image uri for the review. link (str): The url to the online review. source (str): The source of the review. tag (str): The name of the reviewer. text (str): The text of the review. """ TAG = 'Review' def _loadData(self, data): self._data = data self.filter = data.attrib.get('filter') self.id = utils.cast(int, data.attrib.get('id', 0)) self.image = data.attrib.get('image') self.link = data.attrib.get('link') self.source = data.attrib.get('source') self.tag = data.attrib.get('tag') self.text = data.attrib.get('text') class BaseResource(PlexObject): """ Base class for all Art, Banner, Poster, and Theme objects. Attributes: TAG (str): 'Photo' or 'Track' key (str): API URL (/library/metadata/). provider (str): The source of the art or poster, None for Theme objects. ratingKey (str): Unique key identifying the resource. selected (bool): True if the resource is currently selected. thumb (str): The URL to retrieve the resource thumbnail. """ def _loadData(self, data): self._data = data self.key = data.attrib.get('key') self.provider = data.attrib.get('provider') self.ratingKey = data.attrib.get('ratingKey') self.selected = utils.cast(bool, data.attrib.get('selected')) self.thumb = data.attrib.get('thumb') def select(self): key = self._initpath[:-1] data = '%s?url=%s' % (key, quote_plus(self.ratingKey)) try: self._server.query(data, method=self._server._session.put) except xml.etree.ElementTree.ParseError: pass class Art(BaseResource): """ Represents a single Art object. """ TAG = 'Photo' class Banner(BaseResource): """ Represents a single Banner object. """ TAG = 'Photo' class Poster(BaseResource): """ Represents a single Poster object. """ TAG = 'Photo' class Theme(BaseResource): """ Represents a single Theme object. """ TAG = 'Track' @utils.registerPlexObject class Chapter(PlexObject): """ Represents a single Writer media tag. Attributes: TAG (str): 'Chapter' """ TAG = 'Chapter' def _loadData(self, data): self._data = data self.id = utils.cast(int, data.attrib.get('id', 0)) self.filter = data.attrib.get('filter') # I couldn't filter on it anyways self.tag = data.attrib.get('tag') self.title = self.tag self.index = utils.cast(int, data.attrib.get('index')) self.start = utils.cast(int, data.attrib.get('startTimeOffset')) self.end = utils.cast(int, data.attrib.get('endTimeOffset')) @utils.registerPlexObject class Marker(PlexObject): """ Represents a single Marker media tag. Attributes: TAG (str): 'Marker' """ TAG = 'Marker' def __repr__(self): name = self._clean(self.firstAttr('type')) start = utils.millisecondToHumanstr(self._clean(self.firstAttr('start'))) end = utils.millisecondToHumanstr(self._clean(self.firstAttr('end'))) offsets = '%s-%s' % (start, end) return '<%s>' % ':'.join([self.__class__.__name__, name, offsets]) def _loadData(self, data): self._data = data self.id = utils.cast(int, data.attrib.get('id')) self.type = data.attrib.get('type') self.start = utils.cast(int, data.attrib.get('startTimeOffset')) self.end = utils.cast(int, data.attrib.get('endTimeOffset')) @utils.registerPlexObject class Field(PlexObject): """ Represents a single Field. Attributes: TAG (str): 'Field' """ TAG = 'Field' def _loadData(self, data): self._data = data self.name = data.attrib.get('name') self.locked = utils.cast(bool, data.attrib.get('locked')) @utils.registerPlexObject class SearchResult(PlexObject): """ Represents a single SearchResult. Attributes: TAG (str): 'SearchResult' """ TAG = 'SearchResult' def __repr__(self): name = self._clean(self.firstAttr('name')) score = self._clean(self.firstAttr('score')) return '<%s>' % ':'.join([p for p in [self.__class__.__name__, name, score] if p]) def _loadData(self, data): self._data = data self.guid = data.attrib.get('guid') self.lifespanEnded = data.attrib.get('lifespanEnded') self.name = data.attrib.get('name') self.score = utils.cast(int, data.attrib.get('score')) self.year = data.attrib.get('year') @utils.registerPlexObject class Agent(PlexObject): """ Represents a single Agent. Attributes: TAG (str): 'Agent' """ TAG = 'Agent' def __repr__(self): uid = self._clean(self.firstAttr('shortIdentifier')) return '<%s>' % ':'.join([p for p in [self.__class__.__name__, uid] if p]) def _loadData(self, data): self._data = data self.hasAttribution = data.attrib.get('hasAttribution') self.hasPrefs = data.attrib.get('hasPrefs') self.identifier = data.attrib.get('identifier') self.name = data.attrib.get('name') self.primary = data.attrib.get('primary') self.shortIdentifier = self.identifier.rsplit('.', 1)[1] if 'mediaType' in self._initpath: self.languageCodes = self.listAttrs(data, 'code', etag='Language') self.mediaTypes = [] else: self.languageCodes = [] self.mediaTypes = self.findItems(data, cls=AgentMediaType) @property @deprecated('use "languageCodes" instead') def languageCode(self): return self.languageCodes def settings(self): key = '/:/plugins/%s/prefs' % self.identifier data = self._server.query(key) return self.findItems(data, cls=settings.Setting) @deprecated('use "settings" instead') def _settings(self): return self.settings() class AgentMediaType(Agent): """ Represents a single Agent MediaType. Attributes: TAG (str): 'MediaType' """ TAG = 'MediaType' def __repr__(self): uid = self._clean(self.firstAttr('name')) return '<%s>' % ':'.join([p for p in [self.__class__.__name__, uid] if p]) def _loadData(self, data): self.languageCodes = self.listAttrs(data, 'code', etag='Language') self.mediaType = utils.cast(int, data.attrib.get('mediaType')) self.name = data.attrib.get('name') @property @deprecated('use "languageCodes" instead') def languageCode(self): return self.languageCodes