Skip to content

Commit

Permalink
OCP skill keyword matching utils
Browse files Browse the repository at this point in the history
ocp-nlp bus api
  • Loading branch information
JarbasAl committed Jan 6, 2024
1 parent 1483500 commit 776b3d6
Show file tree
Hide file tree
Showing 6 changed files with 122 additions and 146 deletions.
132 changes: 3 additions & 129 deletions ovos_workshop/decorators/ocp.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
# backwards compat imports
from ovos_utils.ocp import MediaType, PlayerState, MediaState, MatchConfidence,\
PlaybackType, PlaybackMode, LoopState, TrackState


def ocp_search():
Expand Down Expand Up @@ -124,132 +127,3 @@ def real_decorator(func):

return real_decorator


try:
from ovos_plugin_common_play.ocp.status import MediaType, PlayerState, \
MediaState, MatchConfidence, PlaybackType, PlaybackMode, LoopState, \
TrackState
except ImportError:

# TODO - manually keep these in sync as needed
# apps interfacing with OCP need the enums,
# but they are native to OCP does not make sense for OCP to import them from here,
# therefore we duplicate them when needed
from enum import IntEnum


class MatchConfidence(IntEnum):
EXACT = 95
VERY_HIGH = 90
HIGH = 80
AVERAGE_HIGH = 70
AVERAGE = 50
AVERAGE_LOW = 30
LOW = 15
VERY_LOW = 1


class TrackState(IntEnum):
DISAMBIGUATION = 1 # media result, not queued for playback

PLAYING_SKILL = 20 # Skill is handling playback internally
PLAYING_AUDIOSERVICE = 21 # Skill forwarded playback to audio service
PLAYING_VIDEO = 22 # Skill forwarded playback to gui player
PLAYING_AUDIO = 23 # Skill forwarded audio playback to gui player
PLAYING_MPRIS = 24 # External media player is handling playback
PLAYING_WEBVIEW = 25 # Media playback handled in browser (eg. javascript)

QUEUED_SKILL = 30 # Waiting playback to be handled inside skill
QUEUED_AUDIOSERVICE = 31 # Waiting playback in audio service
QUEUED_VIDEO = 32 # Waiting playback in gui
QUEUED_AUDIO = 33 # Waiting playback in gui
QUEUED_WEBVIEW = 34 # Waiting playback in gui


class MediaState(IntEnum):
# https://doc.qt.io/qt-5/qmediaplayer.html#MediaStatus-enum
# The status of the media cannot be determined.
UNKNOWN = 0
# There is no current media. PlayerState == STOPPED
NO_MEDIA = 1
# The current media is being loaded. The player may be in any state.
LOADING_MEDIA = 2
# The current media has been loaded. PlayerState== STOPPED
LOADED_MEDIA = 3
# Playback of the current media has stalled due to
# insufficient buffering or some other temporary interruption.
# PlayerState != STOPPED
STALLED_MEDIA = 4
# The player is buffering data but has enough data buffered
# for playback to continue for the immediate future.
# PlayerState != STOPPED
BUFFERING_MEDIA = 5
# The player has fully buffered the current media. PlayerState != STOPPED
BUFFERED_MEDIA = 6
# Playback has reached the end of the current media. PlayerState == STOPPED
END_OF_MEDIA = 7
# The current media cannot be played. PlayerState == STOPPED
INVALID_MEDIA = 8


class PlayerState(IntEnum):
# https://doc.qt.io/qt-5/qmediaplayer.html#State-enum
STOPPED = 0
PLAYING = 1
PAUSED = 2


class LoopState(IntEnum):
NONE = 0
REPEAT = 1
REPEAT_TRACK = 2


class PlaybackType(IntEnum):
SKILL = 0 # skills handle playback whatever way they see fit,
# eg spotify / mycroft common play
VIDEO = 1 # Video results
AUDIO = 2 # Results should be played audio only
AUDIO_SERVICE = 3 # Results should be played without using the GUI
MPRIS = 4 # External MPRIS compliant player
WEBVIEW = 5 # GUI webview, render a url instead of media player
UNDEFINED = 100 # data not available, hopefully status will be updated soon..


class PlaybackMode(IntEnum):
AUTO = 0 # play each entry as considered appropriate,
# ie, make it happen the best way possible
AUDIO_ONLY = 10 # only consider audio entries
VIDEO_ONLY = 20 # only consider video entries
FORCE_AUDIO = 30 # cast video to audio unconditionally
# (audio can still play in mycroft-gui)
FORCE_AUDIOSERVICE = 40 # cast everything to audio service backend,
# mycroft-gui will not be used
EVENTS_ONLY = 50 # only emit ocp events, do not display or play anything.
# allows integration with external interfaces


class MediaType(IntEnum):
GENERIC = 0
AUDIO = 1
MUSIC = 2
VIDEO = 3
AUDIOBOOK = 4
GAME = 5
PODCAST = 6
RADIO = 7
NEWS = 8
TV = 9
MOVIE = 10
TRAILER = 11
VISUAL_STORY = 13
BEHIND_THE_SCENES = 14
DOCUMENTARY = 15
RADIO_THEATRE = 16
SHORT_FILM = 17
SILENT_MOVIE = 18
BLACK_WHITE_MOVIE = 20
CARTOON = 21

ADULT = 69
HENTAI = 70
2 changes: 1 addition & 1 deletion ovos_workshop/skills/active.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

class ActiveSkill(OVOSSkill):
def bind(self, bus):
super(ActiveSkill, self).bind(bus)
super().bind(bus)
if bus:
""" insert skill in active skill list on load """
self.make_active()
Expand Down
122 changes: 112 additions & 10 deletions ovos_workshop/skills/common_play.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,14 @@
from ovos_workshop.skills.ovos import OVOSSkill
from ovos_bus_client import Message
from ovos_utils.log import LOG
from ovos_utils import camel_case_split
from typing import List
from ocp_nlp.features import KeywordFeatures
# backwards compat imports, do not delete, skills import from here


# backwards compat imports, do not delete, skills import from here
from ovos_workshop.decorators.ocp import ocp_play, ocp_next, ocp_pause, ocp_resume, ocp_search, \
ocp_previous, ocp_featured_media, MediaType, MediaState, MatchConfidence, \
PlaybackType, PlaybackMode, PlayerState, LoopState, TrackState
from ocp_nlp.constants import PlayerState, PlaybackType, LoopState, TrackState, \
PlaybackMode, MediaType, MediaState, OCP_ENTITIES


def get_non_properties(obj):
Expand Down Expand Up @@ -50,10 +52,14 @@ def ...
vocab for starting playback is needed.
"""

def __init__(self, name=None, bus=None, **kwargs):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# NOTE: derived skills will likely want to override this list
self.supported_media = [MediaType.GENERIC,
MediaType.AUDIO]
self.supported_media = [MediaType.GENERIC]
skill_name = camel_case_split(self.__class__.__name__)
alt = skill_name.replace(" skill", "").replace(" Skill", "")
self.skill_aliases = [skill_name, alt]

self._search_handlers = [] # added via decorators
self._featured_handlers = [] # added via decorators
self._current_query = None
Expand All @@ -68,7 +74,8 @@ def __init__(self, name=None, bus=None, **kwargs):
self.skill_icon = \
"https://github.com/OpenVoiceOS/ovos-ocp-audio-plugin/raw/master/" \
"ovos_plugin_common_play/ocp/res/ui/images/ocp.png"
OVOSSkill.__init__(self, name, bus, **kwargs)

self.ocp_matchers = {l: KeywordFeatures(l) for l in self.native_langs}

def bind(self, bus):
"""Overrides the normal bind method.
Expand Down Expand Up @@ -104,15 +111,109 @@ def bind(self, bus):
self.add_event("mycroft.stop",
self.__handle_stop_search)

def __handle_ocp_skills_get(self, message):
def register_media_type(self, media_type: MediaType):
""" helper instead of editing self.supported_media directly
will auto-sync changes via bus"""
if media_type not in self.supported_media:
self.supported_media.append(media_type)
LOG.info(f"{self.skill_id} registered type {media_type}")
self.__handle_ocp_skills_get()

def __handle_ocp_skills_get(self, message=None):
""" report skill OCP info
thumbnail and featured tracks inform the OCP homescreen
media_type and skill_name help the classifier disambiguate between media_types
eg, if OCP finds the name of a movie skill in user utterance
it will search netflix instead of spotify
"""
message = message or Message("")
self.bus.emit(
message.reply('ovos.common_play.announce',
{"skill_id": self.skill_id,
"skill_name": self.name,
"skill_name": self.skill_name,
"aliases": self.skill_aliases,
"thumbnail": self.skill_icon,
"media_type": self.supported_media,
"featured_tracks": len(self._featured_handlers) >= 1}))

def ocp_voc_match(self, utterance, lang=None):
"""uses Aho–Corasick algorithm to match OCP keywords
this efficiently matches many keywords against an utterance
OCP keywords are registered via self.register_ocp_keyword
example usages
print(self.ocp_voc_match("play metallica"))
# {'album_name': 'Metallica', 'artist_name': 'Metallica'}
print(self.ocp_voc_match("play the beatles"))
# {'album_name': 'The Beatles', 'series_name': 'The Beatles',
# 'artist_name': 'The Beatles', 'movie_name': 'The Beatles'}
print(self.ocp_voc_match("play rob zombie"))
# {'artist_name': 'Rob Zombie', 'album_name': 'Zombie',
# 'book_name': 'Zombie', 'game_name': 'Zombie', 'movie_name': 'Zombie'}
print(self.ocp_voc_match("play horror movie"))
# {'film_genre': 'Horror', 'cartoon_genre': 'Horror', 'anime_genre': 'Horror',
# 'radio_drama_genre': 'horror', 'video_genre': 'horror',
# 'book_genre': 'Horror', 'movie_name': 'Horror Movie'}
print(self.ocp_voc_match("play science fiction"))
# {'film_genre': 'Science Fiction', 'cartoon_genre': 'Science Fiction',
# 'podcast_genre': 'Fiction', 'anime_genre': 'Science Fiction',
# 'documentary_genre': 'Science', 'book_genre': 'Science Fiction',
# 'artist_name': 'Fiction', 'tv_channel': 'Science',
# 'album_name': 'Science Fiction', 'short_film_name': 'Science',
# 'book_name': 'Science Fiction', 'movie_name': 'Science Fiction'}
"""
lang = lang or self.lang
if lang not in self.ocp_matchers:
return {}
return self.ocp_matchers[lang].extract(utterance)

def register_ocp_keyword(self, label: str, samples: List, langs: List[str] = None):
""" register strings as native OCP keywords (eg, movie_name, artist_name ...)
ocp keywords can be efficiently matched with self.ocp_match helper method
that uses Aho–Corasick algorithm
if the label is a valid OCP entity known by the classifier it will help
the classifier disambiguate between media_types
a full list can be found in ocp_nlp.constants.OCP_ENTITIES
eg, if OCP finds a movie name in user utterances it will
prefer to search netflix instead of spotify
"""
langs = langs or self.native_langs
for l in langs:
if l not in self.ocp_matchers:
self.ocp_matchers[l] = KeywordFeatures(l)
self.ocp_matchers[l].register_entity(label, samples)

self.bus.emit(
Message('ovos.common_play.register_keyword',
{"skill_id": self.skill_id,
"label": label, # if in OCP_ENTITIES it influences classifier
"langs": langs,
"samples": samples}))

def deregister_ocp_keyword(self, media_type: MediaType, label: str,
langs: List[str] = None):
langs = langs or self.native_langs
for l in langs:
if l in self.ocp_matchers:
self.ocp_matchers[l].deregister_entity(label)
self.bus.emit(
Message('ovos.common_play.deregister_keyword',
{"skill_id": self.skill_id,
"label": label,
"langs": langs,
"media_type": media_type}))

def _register_decorated(self):
# register search handlers
for attr_name in get_non_properties(self):
Expand Down Expand Up @@ -162,6 +263,7 @@ def _register_decorated(self):
LOG.warning("multiple declarations of resume playback"
"handler, replacing previous handler")
self.__resume_handler = method

super()._register_decorated()

# volunteer info to OCP
Expand Down
4 changes: 2 additions & 2 deletions ovos_workshop/skills/fallback.py
Original file line number Diff line number Diff line change
Expand Up @@ -232,7 +232,7 @@ def _remove_registered_handler(cls, wrapper_to_del: callable) -> bool:
del cls.fallback_handlers[priority]

if not found_handler:
LOG.warning('No fallback matching {}'.format(wrapper_to_del))
LOG.warning(f'No fallback matching {wrapper_to_del}')
return found_handler

@classmethod
Expand Down Expand Up @@ -440,7 +440,7 @@ def remove_fallback(self, handler_to_del: Optional[callable] = None) -> bool:
del self._fallback_handlers[i]

if not found_handler:
LOG.warning('No fallback matching {}'.format(handler_to_del))
LOG.warning(f'No fallback matching {handler_to_del}')
if len(self._fallback_handlers) == 0:
self.bus.emit(Message("ovos.skills.fallback.deregister",
{"skill_id": self.skill_id}))
Expand Down
4 changes: 2 additions & 2 deletions requirements/requirements.txt
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
ovos-utils < 0.2.0, >=0.0.37
ovos-utils >= 0.1.0a7, < 0.2.0
ovos-bus-client < 0.1.0, >=0.0.9a2
ovos_config < 0.2.0,>=0.0.12
ovos_bus_client < 0.2.0, >=0.0.8
ovos_backend_client < 0.2.0, >=0.1.0
ovos-lingua-franca~=0.4, >=0.4.6
rapidfuzz
4 changes: 2 additions & 2 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ def package_files(directory):

def required(requirements_file):
""" Read requirements file and remove comments and empty lines. """
with open(os.path.join(BASEDIR, requirements_file), 'r') as f:
with open(os.path.join(BASEDIR, requirements_file)) as f:
requirements = f.read().splitlines()
if 'MYCROFT_LOOSE_REQUIREMENTS' in os.environ:
print('USING LOOSE REQUIREMENTS!')
Expand All @@ -50,7 +50,7 @@ def required(requirements_file):


def get_description():
with open(os.path.join(BASEDIR, "README.md"), "r") as f:
with open(os.path.join(BASEDIR, "README.md")) as f:
long_description = f.read()
return long_description

Expand Down

0 comments on commit 776b3d6

Please sign in to comment.