Skip to content
This repository was archived by the owner on May 26, 2022. It is now read-only.

Commit cec39c5

Browse files
committed
rapidfuzz, title caching, cleanup
1 parent 486d9c9 commit cec39c5

File tree

5 files changed

+50
-50
lines changed

5 files changed

+50
-50
lines changed

custom_components/plex_assistant/__init__.py

+11-8
Original file line numberDiff line numberDiff line change
@@ -39,8 +39,10 @@
3939
async def async_setup(hass: HomeAssistant, config: Config):
4040
if DOMAIN in config:
4141
changes_url = "https://github.com/maykar/plex_assistant/blob/master/ver_one_update.md"
42-
message = "Configuration is now handled in the UI, please read the %s for how to migrate " \
43-
"to the new version and more info.%s "
42+
message = (
43+
"Configuration is now handled in the UI, please read the %s for how to migrate "
44+
"to the new version and more info.%s "
45+
)
4446
service_data = {
4547
"title": "Plex Assistant Breaking Changes",
4648
"message": message % (f"[change log]({changes_url})", "."),
@@ -71,17 +73,17 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry):
7173
def pa_executor(_server, start_script_keys):
7274
_pa = PlexAssistant(_server, start_script_keys)
7375
get_devices(hass, _pa)
76+
_LOGGER.debug(f"Media titles: {len(_pa.media['all_titles'])}")
7477
return _pa
7578

7679
pa = await hass.async_add_executor_job(pa_executor, server, list(start_script.keys()))
7780

78-
ifttt_listener = await listeners(hass)
79-
hass.data[DOMAIN][entry.entry_id] = {"remove_listener": ifttt_listener}
80-
8181
tts_dir = hass.config.path() + "/www/plex_assist_tts/"
8282
if tts_errors and not os.path.exists(tts_dir):
8383
os.makedirs(tts_dir, mode=0o777)
8484

85+
ifttt_listener = await listeners(hass)
86+
hass.data[DOMAIN][entry.entry_id] = {"remove_listener": ifttt_listener}
8587
entry.add_update_listener(async_reload_entry)
8688

8789
def handle_input(call):
@@ -108,7 +110,8 @@ def handle_input(call):
108110
return
109111

110112
if pa.media["updated"] < pa.library.search(sort="addedAt:desc", limit=1)[0].addedAt:
111-
pa.update_libraries()
113+
type(pa).media.fget.cache_clear()
114+
_LOGGER.debug(f"Updated Library: {pa.media['updated']}")
112115

113116
device = fuzzy(command["device"] or default_device, pa.device_names)
114117
device = run_start_script(hass, pa, command, start_script, device, default_device)
@@ -135,12 +138,12 @@ def handle_input(call):
135138
play_tts_error(hass, tts_dir, device["entity_id"], error, lang)
136139
return
137140

138-
_LOGGER.debug("Media: %s", str(media))
141+
_LOGGER.debug("Media: %s", str(media.items))
139142

140143
payload = '%s{"playqueue_id": %s, "type": "%s"}' % (
141144
"plex://" if device["device_type"] in ["cast", "sonos"] else "",
142145
media.playQueueID,
143-
media.playQueueType
146+
media.playQueueType,
144147
)
145148

146149
media_service(hass, device["entity_id"], "play_media", payload)

custom_components/plex_assistant/config_flow.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ def get_devices(_self):
1414
if "plex" in info or "cast" in info:
1515
try:
1616
devices.append(_self.hass.states.get(entity.entity_id).attributes.get("friendly_name"))
17-
except:
17+
except AttributeError:
1818
continue
1919
else:
2020
continue
@@ -24,7 +24,7 @@ def get_devices(_self):
2424
def get_servers(_self):
2525
try:
2626
return [x.title for x in _self.hass.config_entries.async_entries("plex")]
27-
except:
27+
except (KeyError, AttributeError):
2828
return []
2929

3030

custom_components/plex_assistant/helpers.py

+26-30
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
import re
22
import time
33
import uuid
4-
import json
54
import pychromecast
65

7-
from fuzzywuzzy import fuzz
8-
from fuzzywuzzy import process
6+
from rapidfuzz import fuzz, process
97
from gtts import gTTS
8+
from json import JSONDecodeError, loads
109
from homeassistant.components.plex.services import get_plex_server
10+
from homeassistant.exceptions import HomeAssistantError, ServiceNotFound
1111
from homeassistant.core import Context
1212
from pychromecast.controllers.plex import PlexController
1313

@@ -16,38 +16,34 @@
1616

1717
def fuzzy(media, lib, scorer=fuzz.QRatio):
1818
if isinstance(lib, list) and len(lib) > 0:
19-
return process.extractOne(media, lib, scorer=scorer)
19+
return process.extractOne(media, lib, scorer=scorer) or ["", 0]
2020
return ["", 0]
2121

2222

23-
def process_config_item(options, item_type):
24-
item = options.get(item_type)
25-
if item:
26-
try:
27-
item = json.loads("{" + item + "}")
28-
for i in item.keys():
29-
_LOGGER.debug(f"{item_type} {i}: {item[i]}")
30-
except Exception:
31-
item = {}
32-
return item
33-
return {}
23+
def process_config_item(options, option_type):
24+
option = options.get(option_type)
25+
if not option:
26+
return {}
27+
try:
28+
option = loads("{" + option + "}")
29+
for i in option.keys():
30+
_LOGGER.debug(f"{option_type} {i}: {option[i]}")
31+
except (TypeError, AttributeError, KeyError, JSONDecodeError):
32+
_LOGGER.warning(f"There is a formatting error in the {option_type.replace('_', ' ')} config.")
33+
option = {}
34+
return option
3435

3536

3637
async def get_server(hass, config, server_name):
3738
try:
3839
await hass.helpers.discovery.async_discover(None, None, "plex", config)
3940
return get_plex_server(hass, server_name)._plex_server
40-
except Exception as ex:
41-
if ex.args[0] == "No Plex servers available":
42-
server_name_str = ", the server_name is correct," if server_name else ""
43-
_LOGGER.warning(
44-
"Plex Assistant: Plex server not found. Ensure that you've setup the HA "
45-
f"Plex integration{server_name_str} and the server is reachable. "
46-
)
47-
else:
48-
template = "An exception of type {0} occurred. Arguments:\n{1!r}"
49-
message = template.format(type(ex).__name__, ex.args)
50-
_LOGGER.warning(message)
41+
except HomeAssistantError as error:
42+
server_name_str = ", the server_name is correct," if server_name else ""
43+
_LOGGER.warning(
44+
f"Plex Assistant: {error.args[0]}. Ensure that you've setup the HA "
45+
f"Plex integration{server_name_str} and the server is reachable. "
46+
)
5147

5248

5349
def get_devices(hass, pa):
@@ -58,7 +54,7 @@ def get_devices(hass, pa):
5854
continue
5955
try:
6056
name = hass.states.get(entity.entity_id).attributes.get("friendly_name")
61-
except Exception:
57+
except AttributeError:
6258
continue
6359
pa.devices[name] = {"entity_id": entity.entity_id, "device_type": dev_type}
6460

@@ -81,7 +77,7 @@ def ifttt_webhook_callback(event):
8177
listener = hass.bus.async_listen("ifttt_webhook_received", ifttt_webhook_callback)
8278
try:
8379
await hass.services.async_call("conversation", "process", {"text": "tell plex to initialize_plex_intent"})
84-
except Exception:
80+
except ServiceNotFound:
8581
pass
8682
return listener
8783

@@ -253,14 +249,14 @@ def filter_media(pa, command, media, library):
253249
media = unwatched if unwatched and not command["random"] else media.episodes()[:30]
254250
elif getattr(media, "TYPE", None) == "episode":
255251
episodes = media.show().episodes()
256-
episodes = episodes[episodes.index(media):episodes.index(media) + 30]
252+
episodes = episodes[episodes.index(media) : episodes.index(media) + 30]
257253
media = pa.server.createPlayQueue(episodes, shuffle=int(command["random"]))
258254
elif getattr(media, "TYPE", None) in ["artist", "album"]:
259255
tracks = media.tracks()
260256
media = pa.server.createPlayQueue(tracks, shuffle=int(command["random"]))
261257
elif getattr(media, "TYPE", None) == "track":
262258
tracks = media.album().tracks()
263-
tracks = tracks[tracks.index(media):]
259+
tracks = tracks[tracks.index(media) :]
264260
media = pa.server.createPlayQueue(tracks, shuffle=int(command["random"]))
265261

266262
if getattr(media, "TYPE", None) != "playqueue" and media:

custom_components/plex_assistant/manifest.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@
99
"requirements": [
1010
"gTTs>=2.2.1",
1111
"pychromecast>=8.0.0",
12-
"fuzzywuzzy==0.18.0",
12+
"rapidfuzz==1.1.1",
1313
"plexapi>=4.3.0",
1414
"awesomeversion>=21.2.2"
1515
]

custom_components/plex_assistant/plex_assistant.py

+10-9
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,13 @@
11
from datetime import datetime
2+
from functools import lru_cache
23

34

45
class PlexAssistant:
56
def __init__(self, server, start_script_keys):
67
self.server = server
78
self.library = self.server.library
89
self.devices = {}
9-
self.media = {}
1010
self.start_script_keys = start_script_keys
11-
self.update_libraries()
1211
self.tv_id = self.get_section_id("show")
1312
self.movie_id = self.get_section_id("movie")
1413
self.music_id = self.get_section_id("artist")
@@ -30,14 +29,16 @@ def section_id(self):
3029
"track": self.music_id,
3130
}
3231

33-
def update_libraries(self):
34-
self.library.reload()
35-
self.media["all_titles"] = []
32+
@property
33+
@lru_cache()
34+
def media(self):
35+
media_items = {"all_titles": []}
3636
for item in ["show", "movie", "artist", "album", "track"]:
37-
self.media[f"{item}_titles"] = [x.title for x in self.library.search(libtype=item, sort="addedAt:desc")]
38-
self.media["all_titles"] += self.media[f"{item}_titles"]
39-
self.media["playlist_titles"] = [x.title for x in self.server.playlists()]
40-
self.media["updated"] = datetime.now()
37+
media_items[f"{item}_titles"] = [x.title for x in self.library.search(libtype=item, sort="addedAt:desc")]
38+
media_items["all_titles"] += media_items[f"{item}_titles"]
39+
media_items["playlist_titles"] = [x.title for x in self.server.playlists()]
40+
media_items["updated"] = datetime.now()
41+
return media_items
4142

4243
def get_section_id(self, section):
4344
section = self.library.search(libtype=section, limit=1)

0 commit comments

Comments
 (0)