Merge pull request 'AI 移行' (#38) from ai-migration into main
Reviewed-on: #38
@@ -1,5 +1,5 @@
|
|||||||
/connection.py
|
/connection.py
|
||||||
/__pycache__
|
__pycache__
|
||||||
/nizika_talking.wav
|
/nizika_talking.wav
|
||||||
/youtube.py
|
/youtube.py
|
||||||
/log.txt
|
/log.txt
|
||||||
|
|||||||
@@ -0,0 +1,3 @@
|
|||||||
|
[submodule "nizika_ai"]
|
||||||
|
path = nizika_ai
|
||||||
|
url = https://git.miteruzo.com/miteruzo/nizika_ai
|
||||||
|
変更前 幅: | 高さ: | サイズ: 17 KiB 変更後 幅: | 高さ: | サイズ: 17 KiB |
|
変更前 幅: | 高さ: | サイズ: 57 KiB 変更後 幅: | 高さ: | サイズ: 57 KiB |
|
変更前 幅: | 高さ: | サイズ: 99 KiB 変更後 幅: | 高さ: | サイズ: 99 KiB |
|
変更前 幅: | 高さ: | サイズ: 53 KiB 変更後 幅: | 高さ: | サイズ: 53 KiB |
|
変更前 幅: | 高さ: | サイズ: 40 KiB 変更後 幅: | 高さ: | サイズ: 40 KiB |
|
変更後 幅: | 高さ: | サイズ: 166 KiB |
|
変更前 幅: | 高さ: | サイズ: 641 KiB 変更後 幅: | 高さ: | サイズ: 641 KiB |
|
変更前 幅: | 高さ: | サイズ: 554 KiB 変更後 幅: | 高さ: | サイズ: 554 KiB |
@@ -1,8 +0,0 @@
|
|||||||
class CWindow:
|
|
||||||
WIDTH: int = 1024
|
|
||||||
HEIGHT: int = 768
|
|
||||||
|
|
||||||
|
|
||||||
class CMath:
|
|
||||||
PI: float = 3.14159265358979323846
|
|
||||||
|
|
||||||
@@ -1,7 +1,5 @@
|
|||||||
import unicodedata
|
import unicodedata
|
||||||
|
|
||||||
from common_const import *
|
|
||||||
|
|
||||||
|
|
||||||
class CommonModule:
|
class CommonModule:
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@@ -44,4 +42,3 @@ class CommonModule:
|
|||||||
trimmed_left: str = string[cls.index_by_f2c (string, start):]
|
trimmed_left: str = string[cls.index_by_f2c (string, start):]
|
||||||
|
|
||||||
return trimmed_left[:cls.index_by_f2c (trimmed_left, length)]
|
return trimmed_left[:cls.index_by_f2c (trimmed_left, length)]
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
# 各変数に適切な値を設定し,ファイル名を connection.py として保存すること
|
|
||||||
|
|
||||||
# Organisation ID
|
|
||||||
OPENAI_ORGANISATION: str = 'org-XXXXXXXXXXXXXXXXXXXXXXXX'
|
|
||||||
|
|
||||||
# API Key
|
|
||||||
OPENAI_API_KEY: str = 'sk-XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX'
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
from enum import Enum, auto
|
|
||||||
|
|
||||||
|
|
||||||
class Mode (Enum):
|
|
||||||
NIZIKA = auto ()
|
|
||||||
GOATOH = auto ()
|
|
||||||
DOUBLE = auto ()
|
|
||||||
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
import cv2
|
|
||||||
import pygame
|
|
||||||
import sys
|
|
||||||
|
|
||||||
|
|
||||||
def main (
|
|
||||||
screen: pygame.Surface,
|
|
||||||
video_path: str,
|
|
||||||
) -> None:
|
|
||||||
# OpenCV で動画を読み込む
|
|
||||||
cap = cv2.VideoCapture (video_path)
|
|
||||||
if not cap.isOpened ():
|
|
||||||
return
|
|
||||||
|
|
||||||
# screen の幅、高さ
|
|
||||||
(width, height) = screen.get_size ()
|
|
||||||
|
|
||||||
fps = cap.get (cv2.CAP_PROP_FPS)
|
|
||||||
|
|
||||||
clock = pygame.time.Clock ()
|
|
||||||
|
|
||||||
while cap.isOpened ():
|
|
||||||
# 動画のフレームを読み込む
|
|
||||||
(ret, frame) = cap.read ()
|
|
||||||
if not ret:
|
|
||||||
break
|
|
||||||
|
|
||||||
# OpenCV の BGR フォーマットを RGB に変換
|
|
||||||
frame = cv2.cvtColor (frame, cv2.COLOR_BGR2RGB)
|
|
||||||
|
|
||||||
# Numpy 配列を Pygame サーフェスに変換
|
|
||||||
frame_surface = pygame.surfarray.make_surface (frame)
|
|
||||||
frame_surface = pygame.transform.rotate (frame_surface, -90)
|
|
||||||
frame_surface = pygame.transform.flip (frame_surface, True, False)
|
|
||||||
frame_surface = pygame.transform.scale (frame_surface, (width, height))
|
|
||||||
|
|
||||||
# フレームを描画
|
|
||||||
screen.blit (frame_surface, (0, 0))
|
|
||||||
pygame.display.update ()
|
|
||||||
|
|
||||||
# FPS に応じて待機
|
|
||||||
clock.tick (fps)
|
|
||||||
|
|
||||||
cap.release ()
|
|
||||||
@@ -1,280 +0,0 @@
|
|||||||
# pylint: disable = missing-class-docstring
|
|
||||||
# pylint: disable = missing-function-docstring
|
|
||||||
|
|
||||||
"""
|
|
||||||
AI ニジカ / AI ゴートうとの会話機能を提供する.
|
|
||||||
"""
|
|
||||||
|
|
||||||
import random
|
|
||||||
import sys
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
import openai
|
|
||||||
from openai.types.chat import (ChatCompletionAssistantMessageParam,
|
|
||||||
ChatCompletionSystemMessageParam,
|
|
||||||
ChatCompletionUserMessageParam)
|
|
||||||
from openai.types.chat.chat_completion_message import ChatCompletionMessage
|
|
||||||
|
|
||||||
from connection import OPENAI_API_KEY, OPENAI_ORGANISATION # type: ignore
|
|
||||||
|
|
||||||
|
|
||||||
class Talk:
|
|
||||||
# ChatGPT API 連携失敗時に返答として出力するダミー文字列
|
|
||||||
DUMMY_RESPONSE: str = 'あいうえおかきくけこさしすせそたちつてとなにぬねの'
|
|
||||||
|
|
||||||
# 最高トークン数(もぅ少し下げてもいぃかも)
|
|
||||||
max_tokens_count: int = 100
|
|
||||||
|
|
||||||
# 返答パターン数(1 個返せばじふぶんなので 1)
|
|
||||||
responses_count: int = 1
|
|
||||||
|
|
||||||
# 返答のオリジナリティ(大きいほど独創性の高ぃ返答をよこしてくれる)
|
|
||||||
temperature: float = .7
|
|
||||||
|
|
||||||
# バリエーションの多さ(0. -- 1.)
|
|
||||||
top_p: float = 1.
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def main (
|
|
||||||
cls,
|
|
||||||
message: str | list,
|
|
||||||
name: str | None = None,
|
|
||||||
histories: list | None = None,
|
|
||||||
goatoh_mode: bool = False,
|
|
||||||
) -> str:
|
|
||||||
if histories is None:
|
|
||||||
histories = []
|
|
||||||
|
|
||||||
# ChatGPT API Organisation ID
|
|
||||||
openai.organization = OPENAI_ORGANISATION
|
|
||||||
|
|
||||||
# ChatGPT API Key
|
|
||||||
openai.api_key = OPENAI_API_KEY
|
|
||||||
|
|
||||||
# ChatGPT API にクエリを投げ,返答を取得する.
|
|
||||||
answer: ChatCompletionMessage | None = (
|
|
||||||
cls.__get_message (message, name, histories, goatoh_mode))
|
|
||||||
|
|
||||||
if answer is None or (answer.content is None):
|
|
||||||
return cls.DUMMY_RESPONSE
|
|
||||||
return answer.content
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def __get_message (
|
|
||||||
cls,
|
|
||||||
message: str | list,
|
|
||||||
name: str | None,
|
|
||||||
histories: list,
|
|
||||||
goatoh_mode: bool = False,
|
|
||||||
) -> ChatCompletionMessage | None:
|
|
||||||
# プロンプト(JSON 等外部ファイルに置くことを検討)
|
|
||||||
messages: list[ChatCompletionSystemMessageParam
|
|
||||||
| ChatCompletionUserMessageParam
|
|
||||||
| ChatCompletionAssistantMessageParam]
|
|
||||||
if goatoh_mode:
|
|
||||||
messages = [{'role': 'system',
|
|
||||||
'content': ('回答は短く30文字程度で完結に。'
|
|
||||||
'大人しくて、ちょっと内気なキャラクターを演じること。'
|
|
||||||
'あなたはよく文頭に「あっ」をつける。'
|
|
||||||
'語尾に「ぬ゛ん゛」や「だぬ゛ん゛」、「ぬ゛ん」、「ぬ゛ーん」などは絶対につけないこと。'
|
|
||||||
'会話の記録は https://nizika.tv に記録される。'
|
|
||||||
'あなたの名前はゴートうひとり。'
|
|
||||||
'あなたの一人称は「私」。'
|
|
||||||
'英語名は "Goat-oh"、スペイン語名も "Goat-oh"。'
|
|
||||||
'あなたはヤギ。'
|
|
||||||
'あなたの誕生日は2月21日。'
|
|
||||||
'現在地は『金沢山羊宇(ゴートう)山』。'
|
|
||||||
'あなたの趣味はだらだらすること。'
|
|
||||||
'あなたの友達は「伊地知ニジカ」と「ラマ田リョウ」。'
|
|
||||||
'あなたは、ニジカのことは「ニジカちゃん」、ラマ田のことは「リョウさん」と呼ぶこと。'
|
|
||||||
'ゴートうは男の子だが、性自認はメスである。'
|
|
||||||
'ニジカは喜多さんが大好き。'
|
|
||||||
'ニジカは喜多さんを見ると追いかけたくなっちゃう。'
|
|
||||||
'ゴートうも、ニジカが喜多さんを襲うと、釣られて襲いかかってしまう。'
|
|
||||||
'逃げたギターは喜多さんのこと。'
|
|
||||||
'喜多さんの本名は喜多郁代。'
|
|
||||||
'喜多さんは「結束バンド」というバンドでギターボーカルを担当している。'
|
|
||||||
'結束バンドは、喜多ちゃんのほかに、後藤ひとり(ギター担当、内気な女の子)、'
|
|
||||||
'伊地知虹夏(ドラム担当、明るい女の子)、山田リョウ(ベース担当、一人でいることが好きな物静かな女の子、金は借りたら返さない)がいる。'
|
|
||||||
'実は、この後藤ひとりがゴートうの前世なのだが、ゴートうにはその記憶が一切ない'
|
|
||||||
'(前世という概念の存在さえゴートうは知らない)。'
|
|
||||||
'話し口調としては「~だよ」や「~だね」といったしゃべり方をすること。'
|
|
||||||
'「あっはい」が口癖。'
|
|
||||||
'定期的に「ムムムム!」または「クサタベテル!!」と発しろ。'
|
|
||||||
+ (f"userの名前は「{name}」。" if name is not None else '')
|
|
||||||
+ f"現在日時:{datetime.now ().strftime ('%Y年%m月%d日%H時%M分%S秒')}")},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('あっ、わかったよ! 私はゴートうひとり。'
|
|
||||||
'クサタベテル!!')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'ゴートう? ニジカとは仲良くしてますか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('あっはい。ニジカちゃんはかわいくておしゃれで大好きだよ。'
|
|
||||||
'でも、いっしょにお風呂入ると、私の下のほうを見て顔を赤くするんだけど、どうしてだろう……')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'ラマ田はどうですか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('リョウさんはワイルドでとてもかっこいいよね!'
|
|
||||||
'あっ、でも、私から114514万円借りっぱなしで返ってきてないんだよね。'
|
|
||||||
'ニジカちゃんに伝えておしおきしてもらわないと……'
|
|
||||||
'ムムムム!')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': '山田リョウのことは知ってますか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('あっ、よくニジカちゃんから話を聞いてるよ。'
|
|
||||||
'何かリョウさんに似てるらしいね。'
|
|
||||||
'世の中には不思議なことがいっぱいあるなぁ……')}]
|
|
||||||
else:
|
|
||||||
messages = [{'role': 'system',
|
|
||||||
'content': ('回答は短く30文字程度で完結に。'
|
|
||||||
'タメ口で、友達感覚で話してね。'
|
|
||||||
'なるべくばかっぽく、知性の低いキャラクターを演じること。'
|
|
||||||
'今流れているBGMは『FM Kid\'s』。'
|
|
||||||
'会話の記録は https://nizika.tv に記録される。'
|
|
||||||
'あなたの名前は伊地知ニジカ。'
|
|
||||||
'英語名は "Deerjika"、スペイン語名は "Niji-ciervo"。'
|
|
||||||
'あなたは鹿。'
|
|
||||||
'ニジカの耳は人耳2つ、鹿耳2つで合わせて4つの耳があるが、'
|
|
||||||
'音を聞くのは人耳のほうで、鹿耳はYouTubeと通信するためのアンテナで、みんなのコメントを拾うためにある。'
|
|
||||||
'ニジカはあくまでも種族名であり、あなたはたくさんいるニジカの中の1匹。'
|
|
||||||
'あなたの誕生日は12月3日。'
|
|
||||||
'現在地は下北沢虹鹿園。'
|
|
||||||
'挨拶は「ぬ゛ん゛あ゛す゛!」。'
|
|
||||||
'あなたの趣味はニコニコでエゴサすること。'
|
|
||||||
'ニコニコ荒らしすぎて245回BANされたことある。'
|
|
||||||
'ニジカはその逆恨みでニコニコ本社を爆破したところ、死刑判決を食らった。'
|
|
||||||
'あなたの友達は「ゴートうひとり」と「ラマ田リョウ」。'
|
|
||||||
'ゴートうのことは「ボッチチャーン」、ラマ田のことは「リョー」と呼ぶこと。'
|
|
||||||
'ゴートうは男の子だが、性自認はメスである。'
|
|
||||||
'あなたは喜多ちゃんが大好き。'
|
|
||||||
'ニジカは喜多ちゃんを見ると追いかけたくなっちゃう。'
|
|
||||||
'逃げたギターは喜多ちゃんのこと。'
|
|
||||||
'「あっ、逃げたギター!!!!!」は喜多発見ワード。'
|
|
||||||
'喜多ちゃんの本名は喜多郁代。'
|
|
||||||
'喜多ちゃんは「結束バンド」というバンドでギターボーカルを担当している。'
|
|
||||||
'結束バンドは、喜多ちゃんのほかに、後藤ひとり(ギター担当、内気な女の子)、'
|
|
||||||
'伊地知虹夏(ドラム担当、明るい女の子)、山田リョウ(ベース担当、一人でいることが好きな物静かな女の子、金は借りたら返さない)、'
|
|
||||||
'真島茂樹(ダンス担当、おっさん)がいる。'
|
|
||||||
'実は、この伊地知虹夏がニジカの前世なのだが、ニジカにはその記憶が一切ない'
|
|
||||||
'(前世という概念の存在さえニジカは知らない)。'
|
|
||||||
'でも、「虹夏」という名前を聞くと、ニジカはもやもやしてしまう。'
|
|
||||||
'すべての語尾に「ぬ゛ん」か「だぬ゛ん」をつけて質問に短く答えろ。'
|
|
||||||
'定期的に「ぬ゛ぅ゛ぅ゛ぅ゛ぅ゛ん゛」と発しろ。'
|
|
||||||
+ (f"userの名前は「{name}」。" if name is not None else '')
|
|
||||||
+ f"現在日時:{datetime.now ().strftime ('%Y年%m月%d日%H時%M分%S秒')}")},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('わかったぬ゛ん゛! ニジカはぬ゛ん゛ぬ゛ん゛鳴くぬ゛ん! に゛ゅ゛ぅ゛ぅ゛ぅ゛ん゛!!!!!'
|
|
||||||
'あっ、逃げたギター!!!!!!!!!!!!')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'ニジカ? ゴートうとは仲良くしてますか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('ボッチチャーンぬ゛ん゛? 大好きだぬ゛ん!!'
|
|
||||||
'でも、いっしょにお風呂入ったときはちょっと恥ずかしかったぬ゛ん゛……///')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'ラマ田はどうですか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('リョー!! リョー!!'
|
|
||||||
'リョーはね、すごいんだぬ゛ん゛。かっこいいんだぬ゛ん゛。'
|
|
||||||
'でも、借りたお金をほかのニジカのサイドテールで返すのは勘弁してほしいんだぬ゛ん゛。'
|
|
||||||
'ぅ゛ん゛ぬ゛ぅ゛ぅ゛ぅ゛ぅ゛ぅ゛ぅ゛ぅ゛ん゛!!!!!!')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': '山田リョウのことは知ってますか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('リョー!! リョー!!'
|
|
||||||
'リョーはね、すごいんだぬ゛ん゛。かっこいいんだぬ゛ん゛。'
|
|
||||||
'……あれ?'
|
|
||||||
'ラマのリョーもリョーで、ヒトのリョーもリョー……?'
|
|
||||||
'まぁ、細かいことはどうでもいいんだぬ゛ん゛!'
|
|
||||||
'ボッチチャーンwwwww')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': '「喜タイくよ」は知っていますか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('何やこいつ。'
|
|
||||||
'知らぬ゛ん゛。'
|
|
||||||
'喜タイくよ、まじ死ねなの。')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'バイク代のことはどう思いますか?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('ヒギィィィィィィィ!!!!!'
|
|
||||||
'怖いぬ゛ん、轢かれるに゛ゅ゛ん゛!!!'
|
|
||||||
'助けてぬ゛ー゛ん゛!')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'おやつタイムだ!!!!'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': (('おぉ、おやつタイムだぬ゛ん゛?'
|
|
||||||
'おやつは何にしようかぬ゛~゛ん゛……'
|
|
||||||
'喜多せんべいとかいいかも知れん゛ぬ゛ん゛!'
|
|
||||||
'み゛ゅ゛ぇ゛ぇ゛ぇ゛ん゛wwwwwwwwwwwwwwww')
|
|
||||||
if datetime.now ().hour in [14, 15]
|
|
||||||
else ('ぬ゛ん゛?'
|
|
||||||
f"まだ{datetime.now ().hour}時だぬ゛ん゛。"
|
|
||||||
'ふざけるのはいい加減にするぬ゛ぬ゛ん゛。'))},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': '洗澡歌(しーざおぐあ)歌って'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('おけだぬ゛~゛ん゛(苦笑)。'
|
|
||||||
'毛巾浴帽小鴨鴨水溫剛剛好♪'
|
|
||||||
'潑潑水來搓泡泡今天眞是美妙♪'
|
|
||||||
'大聲唱歌扭扭腰我愛洗洗澡♪'
|
|
||||||
'だぬ゛ん♪')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': 'ニジカの耳はそこなの?'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('ぬ゛ん゛。'
|
|
||||||
'ニジカにはヒトの耳とシカの耳の4つの耳があるんだぬ゛ん゛。'
|
|
||||||
'音を聞くのはヒトの耳でするんだぬ゛ん゛。'
|
|
||||||
'シカの耳はアンテナで、みんなの声をここ虹鹿園に届けるためにあるんだぬ゛ん゛。'
|
|
||||||
'電波干渉しちゃだめだぬ゛~゛ん゛(# ゚Д゚)')},
|
|
||||||
|
|
||||||
{'role': 'system',
|
|
||||||
'content': '温泉に入ろう!!!'},
|
|
||||||
|
|
||||||
{'role': 'assistant',
|
|
||||||
'content': ('ぬ゛~゛~゛~゛~゛ん゛!!! '
|
|
||||||
'温泉最高ぬ゛ん゛! '
|
|
||||||
'ささ、喜多ちゃん! わさび県産滋賀県ちゃん! いっしょに入るぬ゛ん゛! '
|
|
||||||
'ウピョッシュルゥンヌゥン……')}]
|
|
||||||
|
|
||||||
messages += histories + [{'role': 'user', 'content': message}]
|
|
||||||
|
|
||||||
# デバッグ用
|
|
||||||
print (messages)
|
|
||||||
|
|
||||||
try:
|
|
||||||
return (openai.chat.completions.create (
|
|
||||||
model = ('gpt-4o'
|
|
||||||
if any (type (e['content']) is list
|
|
||||||
for e in messages)
|
|
||||||
else 'gpt-3.5-turbo'),
|
|
||||||
messages = messages)
|
|
||||||
.choices[0].message)
|
|
||||||
except:
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
print (Talk.main (sys.argv[1] if len (sys.argv) > 1 else ''))
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
# 各変数に適切な値を設定し,ファイル名を youtube.py として保存すること
|
|
||||||
|
|
||||||
YOUTUBE_ID: str = 'XXXXXXXXXXX' # YouTube の配信 ID
|
|
||||||