diff --git a/platypush/plugins/media/search/youtube.py b/platypush/plugins/media/search/youtube.py index 37f019e6fd..091563233b 100644 --- a/platypush/plugins/media/search/youtube.py +++ b/platypush/plugins/media/search/youtube.py @@ -1,9 +1,3 @@ -import re -import urllib.parse -import urllib.request - -import requests - from platypush.context import get_plugin from platypush.plugins.media.search import MediaSearcher @@ -16,67 +10,13 @@ class YoutubeMediaSearcher(MediaSearcher): def search(self, query: str, *_, **__): """ - Performs a YouTube search either using the YouTube API (faster and - recommended, it requires the :mod:`platypush.plugins.google.youtube` - plugin to be configured) or parsing the HTML search results (fallback - slower method) + Performs a YouTube search using the ``youtube`` plugin. """ self.logger.info('Searching YouTube for "%s"', query) - - try: - return self._youtube_search_api(query=query) - except Exception as e: - self.logger.warning( - ( - 'Unable to load the YouTube plugin, ' - 'falling back to HTML parse method: %s' - ), - e, - ) - - return self._youtube_search_html_parse(query=query) - - @staticmethod - def _youtube_search_api(query): - yt = get_plugin('google.youtube') - assert yt, 'YouTube plugin not configured' - return [ - { - 'url': 'https://www.youtube.com/watch?v=' + item['id']['videoId'], - **item.get('snippet', {}), - } - for item in yt.search(query=query).output - if item.get('id', {}).get('kind') == 'youtube#video' - ] - - def _youtube_search_html_parse(self, query): - query = urllib.parse.quote(query) - url = "https://www.youtube.com/results?search_query=" + query - html = requests.get(url, timeout=10).content - results = [] - - while html: - m = re.search( - r'()', - html, - ) - if m: - results.append( - {'url': 'https://www.youtube.com' + m.group(2), 'title': m.group(3)} - ) - - html = html.split(m.group(1))[1] - else: - html = '' - - self.logger.info( - '%d YouTube video results for the search query "%s"', - len(results), - query, - ) - - return results + yt = get_plugin('youtube') + assert yt, 'YouTube plugin not available/configured' + return yt.search(query=query).output # vim:sw=4:ts=4:et: diff --git a/platypush/plugins/youtube/__init__.py b/platypush/plugins/youtube/__init__.py index 8719a57e88..28dbd01a9a 100644 --- a/platypush/plugins/youtube/__init__.py +++ b/platypush/plugins/youtube/__init__.py @@ -1,99 +1,77 @@ -from typing import Collection, Optional, Union -from platypush.plugins import action -from platypush.plugins.google import GooglePlugin +import urllib.parse + +import requests + +from platypush.plugins import Plugin, action -class YoutubePlugin(GooglePlugin): +class YoutubePlugin(Plugin): r""" YouTube plugin. - Requirements: + Unlike other Google plugins, this plugin doesn't rely on the Google API. - 1. Create your Google application, if you don't have one already, on - the `developers console `_. + That's because the official YouTube API has been subject to many changes to + prevent scraping, and it requires the user to tinker with the OAuth layer, + app permissions and app validation in order to get it working. - 2. You may have to explicitly enable your user to use the app if the app - is created in test mode. Go to "OAuth consent screen" and add your user's - email address to the list of authorized users. - - 3. Select the scopes that you want to enable for your application, depending - on the integrations that you want to use. - See https://developers.google.com/identity/protocols/oauth2/scopes - for a list of the available scopes. - - 4. Click on "Credentials", then "Create credentials" -> "OAuth client ID". - - 5 Select "Desktop app", enter whichever name you like, and click "Create". - - 6. Click on the "Download JSON" icon next to your newly created client ID. - - 7. Generate a credentials file for the required scope: - - .. code-block:: bash - - mkdir -p /credentials/google - python -m platypush.plugins.google.credentials \ - 'youtube.readonly' \ - /credentials/google/client_secret.json + Instead, it relies on a `Piped `_. - :param query: Query string (default: empty string) - :param types: List of types to retrieve (default: video). - See the `Getting started - Resources - `_. - :param max_results: Maximum number of items that will be returned (default: 25). - :param kwargs: Any extra arguments that will be transparently passed to the YouTube API. - See the `Getting started - parameters - `_. - :return: A list of YouTube resources. - See the `Getting started - Resource - `_. """ + self.logger.info('Searching YouTube for "%s"', query) + query = urllib.parse.quote(query) + url = f"{self._piped_api_url}/search?q=" + query + "&filter=all" + rs = requests.get(url, timeout=20) + rs.raise_for_status() + results = [ + { + "url": "https://www.youtube.com" + item["url"], + "title": item["title"], + "image": item["thumbnail"], + "duration": item["duration"], + "description": item["shortDescription"], + } + for item in rs.json().get("items", []) + ] - parts = parts or self._default_parts[:] - if isinstance(parts, list): - parts = ','.join(parts) - - types = types or self._default_types[:] - if isinstance(types, list): - types = ','.join(types) - - service = self.get_service('youtube', 'v3') - result = ( - service.search() - .list(part=parts, q=query, type=types, maxResults=max_results, **kwargs) - .execute() + self.logger.info( + '%d YouTube video results for the search query "%s"', + len(results), + query, ) - events = result.get('items', []) - return events + return results # vim:sw=4:ts=4:et: