ニジカ AI 共通サービス
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

415 lines
12 KiB

  1. """
  2. AI ニジカ常時稼動バッチ
  3. """
  4. from __future__ import annotations
  5. import asyncio
  6. import json
  7. import os
  8. import random
  9. import subprocess
  10. from asyncio import Lock
  11. from datetime import date, datetime, time, timedelta
  12. from typing import Any, Callable, TypedDict, cast
  13. import nicolib
  14. import queries_to_answers as q2a
  15. from nicolib import VideoInfo
  16. from nizika_ai.config import DB
  17. from nizika_ai.consts import Character, GPTModel, QueryType
  18. from nizika_ai.models import Query
  19. KIRIBAN_VIEWS_COUNTS: list[int] = sorted ({ *range (1_000, 10_000, 1_000),
  20. *range (10_000, 1_000_001, 10_000),
  21. 114_514, 1_940, 2_450, 5_100,
  22. 19_400, 24_500, 51_000, 93_194, 2_424, 242_424, 1_919,
  23. 4_545, 194_245, 245_194, 510_245 },
  24. reverse = True)
  25. kiriban_list: list[tuple[int, VideoInfo, datetime]] = []
  26. watched_videos: set[str] = set ()
  27. lock = Lock ()
  28. async def main (
  29. ) -> None:
  30. """
  31. メーン処理
  32. """
  33. await asyncio.gather (
  34. queries_to_answers (),
  35. report_kiriban (),
  36. report_nico (),
  37. update_kiriban_list (),
  38. report_snack_time (),
  39. report_hot_spring_time (),
  40. reconnect_db ())
  41. async def queries_to_answers (
  42. ) -> None:
  43. """
  44. クエリ処理
  45. """
  46. while True:
  47. loop = asyncio.get_running_loop ()
  48. await loop.run_in_executor (None, q2a.main)
  49. await asyncio.sleep (10)
  50. async def report_kiriban (
  51. ) -> None:
  52. """
  53. キリ番祝ひ
  54. """
  55. while True:
  56. if not kiriban_list:
  57. await wait_until (time (15, 0))
  58. continue
  59. # キリ番祝ひ
  60. async with lock:
  61. (views_count, video_info, uploaded_at) = (
  62. kiriban_list.pop (random.randint (0, len (kiriban_list) - 1)))
  63. video_code = video_info['contentId']
  64. comments = fetch_comments (video_code)
  65. popular_comments = sorted (comments,
  66. key = lambda c: c['nico_count'],
  67. reverse = True)[:10]
  68. latest_comments = sorted (comments,
  69. key = lambda c: c['posted_at'],
  70. reverse = True)[:10]
  71. prompt = (f"{ _format_elapsed (uploaded_at) }前にニコニコに投稿された"
  72. f"『{ video_info['title'] }』という動画が{ views_count }再生を突破しました。\n"
  73. f"コメント数は{ len (comments) }件です。\n")
  74. if video_info['tags']:
  75. prompt += f"つけられたタグは「{ '」、「'.join (video_info['tags']) }」です。\n"
  76. if comments:
  77. prompt += f"人気のコメントは次の通りです:「{ '」、「'.join (c['content'] for c in popular_comments) }」\n"
  78. if latest_comments != popular_comments:
  79. prompt += f"最新のコメントは次の通りです:「{ '」、「'.join (c['content'] for c in latest_comments) }」\n"
  80. prompt += f"""
  81. 概要には次のように書かれています:
  82. ```html
  83. { video_info['description'] }
  84. ```
  85. このことについて、何かお祝いメッセージを下さい。
  86. ただし、そのメッセージ内には再生数の数値を添えてください。
  87. また、つけられたタグ、コメントからどのような動画か想像し、説明してください。"""
  88. _add_query (prompt, QueryType.KIRIBAN, { 'video_code': video_code })
  89. # 待ち時間計算
  90. dt = datetime.now ()
  91. d = dt.date ()
  92. if dt.hour >= 15:
  93. d += timedelta (days = 1)
  94. remain = max (len (kiriban_list), 1)
  95. td = (datetime.combine (d, time (15, 0)) - dt) / remain
  96. # まれに時刻跨ぎでマイナスになるため
  97. if td.total_seconds () < 0:
  98. td = timedelta (seconds = 0)
  99. await asyncio.sleep (td.total_seconds ())
  100. async def update_kiriban_list (
  101. ) -> None:
  102. """
  103. キリ番リストの更新
  104. """
  105. while True:
  106. await wait_until (time (15, 0))
  107. new_list = fetch_kiriban_list (datetime.now ().date ())
  108. if not new_list:
  109. continue
  110. async with lock:
  111. have = { k[1]['contentId'] for k in kiriban_list }
  112. for item in new_list:
  113. if item[1]['contentId'] not in have:
  114. kiriban_list.append (item)
  115. have.add (item[1]['contentId'])
  116. def fetch_kiriban_list (
  117. base_date: date,
  118. ) -> list[tuple[int, VideoInfo, datetime]]:
  119. """
  120. キリ番を迎へた動画のリストを取得する.
  121. Parameters
  122. ----------
  123. base_date: date
  124. 基準日
  125. Return
  126. ------
  127. list[tuple[int, VideoInfo, datetime]]
  128. 動画リスト(キリ番基準再生数、対象動画情報、投稿日時のタプル)
  129. """
  130. result = subprocess.run (
  131. ['python3', '/root/nizika_nico/get_kiriban_list.py',
  132. str (base_date), *map (str, KIRIBAN_VIEWS_COUNTS)],
  133. cwd = '/root/nizika_nico',
  134. env = os.environ,
  135. capture_output = True,
  136. text = True)
  137. kl: list[list[int | str]]
  138. try:
  139. kl = json.loads (result.stdout)
  140. except Exception:
  141. kl = []
  142. return [(cast (int, k[0]), video_info, str_to_datetime (cast (str, k[2])))
  143. for k in kl
  144. if (video_info := nicolib.fetch_video_info (cast (str, k[1]))) is not None]
  145. def fetch_comments (
  146. video_code: str,
  147. ) -> list[CommentDict]:
  148. """
  149. 動画のコメント・リストを取得する.
  150. Parameters
  151. ----------
  152. video_code: str
  153. ニコニコの動画コード
  154. Return
  155. ------
  156. list[CommentDict]
  157. コメント・リスト
  158. """
  159. result = subprocess.run (
  160. ['python3', 'get_comments_by_video_code.py', video_code],
  161. cwd = '/root/nizika_nico',
  162. env = os.environ,
  163. capture_output = True,
  164. text = True)
  165. rows: list[dict[str, Any]] = json.loads (result.stdout)
  166. comments: list[CommentDict] = []
  167. for row in rows:
  168. row['posted_at'] = str_to_datetime (row['posted_at'])
  169. comments.append (cast (CommentDict, row))
  170. return comments
  171. def fetch_latest_deerjika (
  172. ) -> VideoInfo | None:
  173. """
  174. 最新のぼざクリ動画を取得する.
  175. Return
  176. ------
  177. VideoInfo | None
  178. 動画情報
  179. """
  180. return nicolib.fetch_latest_video (['伊地知ニジカ',
  181. 'ぼざろクリーチャーシリーズ',
  182. 'ぼざろクリーチャーシリーズ外伝'])
  183. async def report_nico (
  184. ) -> None:
  185. """
  186. ニコニコから最新のぼざクリを取得し,まだ報知してゐなかったら報知する.
  187. """
  188. while True:
  189. latest_deerjika = fetch_latest_deerjika ()
  190. if latest_deerjika and latest_deerjika['contentId'] not in watched_videos:
  191. video = latest_deerjika
  192. watched_videos.add (video['contentId'])
  193. prompt = f"""ニコニコに『{ video['title'] }』という動画がアップされました。
  194. つけられたタグは「{ '」、「'.join (video['tags']) }」です。
  195. 概要には次のように書かれています:
  196. ```html
  197. { video['description'] }
  198. ```
  199. このことについて、みんなに告知するとともに、ニジカちゃんの感想を教えてください。"""
  200. _add_query (prompt, QueryType.NICO_REPORT, { 'video_code': video['contentId'] })
  201. await asyncio.sleep (60)
  202. async def wait_until (
  203. t: time,
  204. ) -> None:
  205. """
  206. 指定した時刻まで待つ.
  207. Parameters
  208. ----------
  209. t: time
  210. 次に実行を続行するまでの時刻
  211. """
  212. dt = datetime.now ()
  213. d = dt.date ()
  214. if dt.time () >= t:
  215. d += timedelta (days = 1)
  216. await asyncio.sleep ((datetime.combine (d, t) - dt).total_seconds ())
  217. async def report_snack_time (
  218. ) -> None:
  219. """
  220. おやつタイムを報知する.
  221. """
  222. while True:
  223. await wait_until (time (15, 0))
  224. _add_query ('おやつタイムだ!!!!', QueryType.SNACK_TIME)
  225. async def report_hot_spring_time (
  226. ) -> None:
  227. """
  228. 温泉タイムを報知する.
  229. """
  230. while True:
  231. await wait_until (time (21, 0))
  232. _add_query ('温泉に入ろう!!!', QueryType.HOT_SPRING)
  233. async def reconnect_db (
  234. ) -> None:
  235. while True:
  236. await asyncio.sleep (600)
  237. try:
  238. ensure_mysql_alive ()
  239. except Exception as ex:
  240. if getattr (ex, 'args', [None])[0] not in (2006, 2013):
  241. raise
  242. print (f"[reconnect_db] { type (ex).__name__ }: { ex }")
  243. safe_reconnect ()
  244. def ensure_mysql_alive (
  245. ) -> None:
  246. conn = DB.connection ('mysql').get_connection ()
  247. conn.ping ()
  248. def safe_reconnect (
  249. ) -> None:
  250. try:
  251. DB.reconnect ('mysql')
  252. except Exception as ex:
  253. if getattr (ex, 'args', [None])[0] not in (2006, 2013):
  254. raise
  255. print (f"[safe_reconnect] { type (ex).__name__ }: { ex }")
  256. def run_with_mysql_retry (
  257. fn: Callable[..., Any],
  258. *args,
  259. **kwargs,
  260. ) -> Any:
  261. last = None
  262. for _ in range (2):
  263. try:
  264. ensure_mysql_alive ()
  265. return fn (*args, **kwargs)
  266. except Exception as ex:
  267. if getattr (ex, 'args', [None])[0] not in (2006, 2013):
  268. raise
  269. last = ex
  270. print (f"[run_with_mysql_retry] { type (ex).__name__ }: { ex }")
  271. safe_reconnect ()
  272. if last:
  273. raise last
  274. def _add_query (
  275. content: str,
  276. query_type: QueryType,
  277. transfer_data: dict | None = None,
  278. ) -> None:
  279. query = Query ()
  280. query.user_id = None
  281. query.target_character = Character.DEERJIKA.value
  282. query.content = content
  283. query.query_type = query_type.value
  284. query.model = GPTModel.GPT4_O.value
  285. query.sent_at = datetime.now ()
  286. query.answered = False
  287. if transfer_data is not None:
  288. query.transfer_data = transfer_data
  289. run_with_mysql_retry (query.save)
  290. def _format_elapsed (
  291. uploaded_at: datetime,
  292. ) -> str:
  293. """
  294. 指定した時刻から現在までの時間を見やすぃ文字列に変換する.
  295. Parameters
  296. ----------
  297. uploaded_at: datetime
  298. 基準日時
  299. Return
  300. ------
  301. str
  302. 変換後文字列
  303. """
  304. delta = datetime.now () - uploaded_at
  305. days = delta.days
  306. seconds = delta.seconds
  307. (hours, seconds) = divmod (seconds, 3600)
  308. (mins, seconds) = divmod (seconds, 60)
  309. return f"{ days }日{ hours }時間{ mins }分{ seconds }秒"
  310. def str_to_datetime (
  311. s: str,
  312. ) -> datetime:
  313. formats: list[str] = [
  314. '%Y-%m-%d %H:%M:%S.%f',
  315. '%Y-%m-%d %H:%M:%S']
  316. for f in formats:
  317. try:
  318. return datetime.strptime (s, f)
  319. except ValueError:
  320. pass
  321. raise ValueError ('うんち!w')
  322. class CommentDict (TypedDict):
  323. id: int
  324. video_id: int
  325. comment_no: int
  326. user_id: int
  327. content: str
  328. posted_at: datetime
  329. nico_count: int
  330. vpos_ms: int
  331. kiriban_list = (
  332. fetch_kiriban_list ((now := datetime.now ()).date ()
  333. - timedelta (days = 1 if now.hour < 15 else 0)))
  334. if __name__ == '__main__':
  335. asyncio.run (main ())