From e214ed1b6486d370a44dfa9be772e4a8d61b14b9 Mon Sep 17 00:00:00 2001
From: Christian Heller <c.heller@plomlompom.de>
Date: Sat, 30 Nov 2024 19:40:55 +0100
Subject: [PATCH] Refactor YoutubeQuery creation.

---
 src/ytplom/http.py | 62 +++++-----------------------------------------
 src/ytplom/misc.py | 56 +++++++++++++++++++++++++++++++++++++++++
 2 files changed, 62 insertions(+), 56 deletions(-)

diff --git a/src/ytplom/http.py b/src/ytplom/http.py
index b2366d5..cd66699 100644
--- a/src/ytplom/http.py
+++ b/src/ytplom/http.py
@@ -1,5 +1,4 @@
 """Collect directly HTTP-related elements."""
-from datetime import datetime
 from http.server import HTTPServer, BaseHTTPRequestHandler
 from json import dumps as json_dumps
 from pathlib import Path
@@ -10,17 +9,16 @@ from urllib.request import urlretrieve
 from urllib.error import HTTPError
 from jinja2 import (  # type: ignore
         Environment as JinjaEnv, FileSystemLoader as JinjaFSLoader)
-import googleapiclient.discovery  # type: ignore
 from ytplom.misc import (
-        B64Str, DatetimeStr, FilesWithIndex, FlagName, NotFoundException,
-        PlayerUpdateId, QueryId, QueryText, QuotaCost, UrlStr, YoutubeId,
-        FILE_FLAGS, PATH_APP_DATA, PATH_CACHE, TIMESTAMP_FMT,
-        YOUTUBE_URL_PREFIX,
+        B64Str, FilesWithIndex, FlagName, NotFoundException, PlayerUpdateId,
+        QueryId, QueryText, QuotaCost, UrlStr, YoutubeId,
+        FILE_FLAGS, PATH_APP_DATA, PATH_THUMBNAILS, YOUTUBE_URL_PREFIX,
         ensure_expected_dirs,
         Config, DbConnection, DownloadsManager, Player, QuotaLog, VideoFile,
         YoutubeQuery, YoutubeVideo
 )
 
+# type definitions for mypy
 PageNames: TypeAlias = dict[str, Path]
 ParamsStr = NewType('ParamsStr', str)
 TemplateContext: TypeAlias = dict[
@@ -33,11 +31,8 @@ TemplateContext: TypeAlias = dict[
 ]
 
 # API expectations
-PATH_THUMBNAILS = PATH_CACHE.joinpath('thumbnails')
 THUMBNAIL_URL_PREFIX = UrlStr('https://i.ytimg.com/vi/')
 THUMBNAIL_URL_SUFFIX = UrlStr('/default.jpg')
-QUOTA_COST_YOUTUBE_SEARCH = QuotaCost(100)
-QUOTA_COST_YOUTUBE_DETAILS = QuotaCost(1)
 
 # template paths
 PATH_TEMPLATES = PATH_APP_DATA.joinpath('templates')
@@ -158,53 +153,8 @@ class _TaskHandler(BaseHTTPRequestHandler):
 
     def _receive_yt_query(self, query_txt: QueryText) -> None:
         conn = DbConnection()
-
-        def collect_results(query_txt: QueryText) -> list[YoutubeVideo]:
-            ensure_expected_dirs([PATH_THUMBNAILS])
-            youtube = googleapiclient.discovery.build(
-                    'youtube', 'v3', developerKey=self.server.config.api_key)
-            QuotaLog.update(conn, QUOTA_COST_YOUTUBE_SEARCH)
-            search_request = youtube.search().list(
-                    q=query_txt,
-                    part='snippet',
-                    maxResults=25,
-                    safeSearch='none',
-                    type='video')
-            results: list[YoutubeVideo] = []
-            ids_to_detail: list[YoutubeId] = []
-            for item in search_request.execute()['items']:
-                video_id: YoutubeId = item['id']['videoId']
-                ids_to_detail += [video_id]
-                snippet = item['snippet']
-                urlretrieve(snippet['thumbnails']['default']['url'],
-                            PATH_THUMBNAILS.joinpath(f'{video_id}.jpg'))
-                results += [YoutubeVideo(id_=video_id,
-                                         title=snippet['title'],
-                                         description=snippet['description'],
-                                         published_at=snippet['publishedAt'])]
-            QuotaLog.update(conn, QUOTA_COST_YOUTUBE_DETAILS)
-            ids_for_details = ','.join([r.id_ for r in results])
-            videos_request = youtube.videos().list(id=ids_for_details,
-                                                   part='content_details')
-            unfinished_streams: list[YoutubeId] = []
-            for i, detailed in enumerate(videos_request.execute()['items']):
-                result = results[i]
-                assert result.id_ == detailed['id']
-                content_details: dict[str, str] = detailed['contentDetails']
-                if 'P0D' == content_details['duration']:
-                    unfinished_streams += [result.id_]
-                    continue
-                result.set_duration_from_yt_string(content_details['duration'])
-                result.definition = content_details['definition'].upper()
-            return [r for r in results if r.id_ not in unfinished_streams]
-
-        query_data = YoutubeQuery(
-                None, query_txt,
-                DatetimeStr(datetime.now().strftime(TIMESTAMP_FMT)))
-        query_data.save(conn)
-        for result in collect_results(query_txt):
-            result.save(conn)
-            result.save_to_query(conn, query_data.id_)
+        query_data = YoutubeQuery.new_by_request_saved(
+                conn, self.server.config, query_txt)
         conn.commit_close()
         self._redirect(Path('/')
                        .joinpath(PAGE_NAMES['yt_query'])
diff --git a/src/ytplom/misc.py b/src/ytplom/misc.py
index ca478a1..012c3f5 100644
--- a/src/ytplom/misc.py
+++ b/src/ytplom/misc.py
@@ -8,12 +8,14 @@ from random import shuffle
 from time import time, sleep
 from datetime import datetime, timedelta
 from json import loads as json_loads
+from urllib.request import urlretrieve
 from uuid import uuid4
 from pathlib import Path
 from sqlite3 import connect as sql_connect, Cursor, Row
 from threading import Thread
 from queue import Queue
 # non-included libs
+import googleapiclient.discovery  # type: ignore
 from mpv import MPV  # type: ignore
 from yt_dlp import YoutubeDL  # type: ignore
 
@@ -48,6 +50,7 @@ PATH_CACHE = Path.home().joinpath('.cache/ytplom')
 PATH_DOWNLOADS = Path.home().joinpath('ytplom_downloads')
 PATH_DB = PATH_APP_DATA.joinpath('db.sql')
 PATH_TEMP = PATH_CACHE.joinpath('temp')
+PATH_THUMBNAILS = PATH_CACHE.joinpath('thumbnails')
 PATH_CONFFILE = Path.home().joinpath('.config/ytplom/config.json')
 
 # yt_dlp config
@@ -59,6 +62,8 @@ YT_DL_PARAMS = {'paths': {'home': str(PATH_DOWNLOADS),
 
 # Youtube API expectations
 YOUTUBE_URL_PREFIX = UrlStr('https://www.youtube.com/watch?v=')
+QUOTA_COST_YOUTUBE_SEARCH = QuotaCost(100)
+QUOTA_COST_YOUTUBE_DETAILS = QuotaCost(1)
 
 # database stuff
 EXPECTED_DB_VERSION = 1
@@ -220,6 +225,57 @@ class YoutubeQuery(DbData):
         self.text = QueryText(text)
         self.retrieved_at = retrieved_at
 
+    @classmethod
+    def new_by_request_saved(cls,
+                             conn: DbConnection,
+                             config: Config,
+                             query_txt: QueryText
+                             ) -> Self:
+        """Query YT API and save results."""
+        ensure_expected_dirs([PATH_THUMBNAILS])
+        youtube = googleapiclient.discovery.build(
+                'youtube', 'v3', developerKey=config.api_key)
+        QuotaLog.update(conn, QUOTA_COST_YOUTUBE_SEARCH)
+        search_request = youtube.search().list(
+                q=query_txt,
+                part='snippet',
+                maxResults=25,
+                safeSearch='none',
+                type='video')
+        results: list[YoutubeVideo] = []
+        ids_to_detail: list[YoutubeId] = []
+        for item in search_request.execute()['items']:
+            video_id: YoutubeId = item['id']['videoId']
+            ids_to_detail += [video_id]
+            snippet = item['snippet']
+            urlretrieve(snippet['thumbnails']['default']['url'],
+                        PATH_THUMBNAILS.joinpath(f'{video_id}.jpg'))
+            results += [YoutubeVideo(id_=video_id,
+                                     title=snippet['title'],
+                                     description=snippet['description'],
+                                     published_at=snippet['publishedAt'])]
+        QuotaLog.update(conn, QUOTA_COST_YOUTUBE_DETAILS)
+        ids_for_details = ','.join([r.id_ for r in results])
+        videos_request = youtube.videos().list(id=ids_for_details,
+                                               part='content_details')
+        unfinished_streams: list[YoutubeId] = []
+        for i, detailed in enumerate(videos_request.execute()['items']):
+            result = results[i]
+            assert result.id_ == detailed['id']
+            content_details: dict[str, str] = detailed['contentDetails']
+            if 'P0D' == content_details['duration']:
+                unfinished_streams += [result.id_]
+                continue
+            result.set_duration_from_yt_string(content_details['duration'])
+            result.definition = content_details['definition'].upper()
+        query = cls(None, query_txt,
+                    DatetimeStr(datetime.now().strftime(TIMESTAMP_FMT)))
+        query.save(conn)
+        for result in [r for r in results if r.id_ not in unfinished_streams]:
+            result.save(conn)
+            result.save_to_query(conn, query.id_)
+        return query
+
     @classmethod
     def get_all_for_video(cls,
                           conn: DbConnection,
-- 
2.30.2