#3 クエリの処理

このコミットが含まれているのは:
2024-11-28 02:25:50 +09:00
コミット b244f229f9
6個のファイルの変更380行の追加21行の削除
+20
ファイルの表示
@@ -0,0 +1,20 @@
from __future__ import annotations
import os
from typing import TypedDict
CONFIG: dict[str, DbConfig] = { 'mysql': { 'driver': 'mysql',
'host': 'localhost',
'database': 'nizika_ai',
'user': os.environ['MYSQL_USER'],
'password': os.environ['MYSQL_PASS'],
'prefix': '' } }
class DbConfig (TypedDict):
driver: str
host: str
database: str
user: str
password: str
prefix: str
+8
ファイルの表示
@@ -0,0 +1,8 @@
# 各変数に適切な値を設定し,ファイル名を connection.py として保存すること
# Organisation ID
OPENAI_ORGANISATION: str = 'org-XXXXXXXXXXXXXXXXXXXXXXXX'
# API Key
OPENAI_API_KEY: str = 'sk-XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX'
+1 -18
ファイルの表示
@@ -1,16 +1,8 @@
from __future__ import annotations
import os
from typing import TypedDict
from config import CONFIG
from eloquent import DatabaseManager, Schema
CONFIG: dict[str, DbConfig] = { 'mysql': { 'driver': 'mysql',
'host': 'localhost',
'database': 'nizika_ai',
'user': os.environ['MYSQL_USER'],
'password': os.environ['MYSQL_PASS'],
'prefix': '' } }
DB = DatabaseManager (CONFIG)
SCHEMA = Schema (DB)
@@ -91,14 +83,5 @@ def add_constraints_to_query_answer_histories (
table.foreign ('answer_id').references ('id').on ('answers').on_update ('cascade').on_delete ('cascade')
class DbConfig (TypedDict):
driver: str
host: str
database: str
user: str
password: str
prefix: str
if __name__ == '__main__':
main ()
+15 -3
ファイルの表示
@@ -39,14 +39,14 @@ class Query (Model):
@property
def user (
self,
) -> User:
) -> User | None:
return self.belongs_to (User)
@property
def answer_histories (
self,
) -> list[QueryAnswerHistory]:
return self.has_many (QueryAnswerHistory)
) -> list[Answer]:
return [x.answer for x in self.has_many (QueryAnswerHistory)]
class QueryAnswerHistory (Model):
@@ -54,6 +54,18 @@ class QueryAnswerHistory (Model):
query_id: int
answer_id: int
@property
def query (
self,
) -> Query:
return self.belongs_to (Query)
@property
def answer (
self,
) -> Answer:
return self.belongs_to (Answer)
class User (Model):
id: int
+56
ファイルの表示
@@ -0,0 +1,56 @@
from __future__ import annotations
import random
from datetime import datetime
from enum import Enum
from eloquent import DatabaseManager, Model
from models import Answer, Query, User
from talk import Talk
def main (
) -> None:
queries: list[Query] = Query.where ('answered', False).get ()
query: Query = random.choice (queries)
message: str | list[dict[str, str | dict[str, str]]]
if query.image_url is None:
message = query.content
else:
message = [{ 'type': 'text', 'text': query.content },
{ 'type': 'image_url', 'image_url': query.image_url }]
user: User | None = query.user
user_name: str | None = None
if user is not None:
user_name = user.name
histories: list[dict[str, str]] = []
for history in query.answer_histories:
if history.query is not None:
histories.append ({ 'role': 'user', 'content': history.query.content })
histories.append ({ 'role': 'assistant', 'content': history.content })
if query.target_character & Character.DEERJIKA.value:
answer = Answer ()
answer.query_id = query.id
answer.character = Character.DEERJIKA.value
answer.content = Talk.main (query.content, user_name, histories)
answer.answer_type = query.query_type
answer.sent_at = datetime.now ()
answer.save ()
if query.target_character & Character.GOATOH.value:
answer = Answer ()
answer.query_id = query.id
answer.character = Character.GOATOH.value
answer.content = Talk.main (query.content, user_name, histories, True)
answer.answer_type = query.query_type
answer.sent_at = datetime.now ()
answer.save ()
class Character (Enum):
DEERJIKA = 1
GOATOH = 2
if __name__ == '__main__':
main ()
+280
ファイルの表示
@@ -0,0 +1,280 @@
# 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 ''))