Compare commits

...

41 Commits

Author SHA1 Message Date
  みてるぞ b3ae86033c main.py にリネーム 1 week ago
  みてるぞ 88e710572a 現行の分は移行完了したので main.py 削除 1 week ago
  みてるぞ 9a68a29e1b #32 完了 1 week ago
  みてるぞ 270b4515d8 #35 1 week ago
  みてるぞ 9d0b5aff70 Revert "#35" 1 week ago
  みてるぞ 41f5a7718f #35 1 week ago
  みてるぞ 4ad5868b63 #34 2 weeks ago
  みてるぞ 37c9947d4a #34 動画再生可能に 2 weeks ago
  みてるぞ 6ee5582a32 不要なファイル削除 3 weeks ago
  みてるぞ 12fbdbc7e2 #35 チャット取得確認用出力追加 3 weeks ago
  みてるぞ a9ba0f697e #33 処理落ち対策 3 weeks ago
  みてるぞ 93fc438d8a #31 layer を GameObject の 属性化 3 weeks ago
  みてるぞ 50281f9120 #31 3 weeks ago
  みてるぞ c6028507ea #31 3 weeks ago
  みてるぞ 9149483dcb #31 3 weeks ago
  みてるぞ a7785fa2c1 Merge branch 'ai-migration' of https://git.miteruzo.com/miteruzo/nizika_broadcast into ai-migration 3 weeks ago
  みてるぞ 49661dad71 #31 3 weeks ago
  みてるぞ fb5b64b49b #31 誤字修正 3 weeks ago
  みてるぞ e49eff0876 #31 ニジカのクエリ処理 3 weeks ago
  みてるぞ af862a7981 #31 ぼちぼち 3 weeks ago
  みてるぞ 29b831e380 #31 3 weeks ago
  みてるぞ 98703409ef ぼちぼち ( #31 ) 3 weeks ago
  みてるぞ 7289fe5812 nizika_ai をいったん除外 ( #31 ) 4 weeks ago
  みてるぞ ba5a428f7b #31 4 weeks ago
  みてるぞ 05cd19525c 大月追加 1 month ago
  みてるぞ 80dc32dbb7 #31 tyotto 1 month ago
  みてるぞ 75f3f1cff3 31 ヨヨコぼちぼち 1 month ago
  みてるぞ 2ba68d9e31 #31 き太く陽調整 1 month ago
  みてるぞ 65fe453426 #31 き太移行完了 1 month ago
  みてるぞ 2ce5329aae #31 き太く陽設置(挙動が不安定) 1 month ago
  みてるぞ 20cf2cec04 #31 吹き出しの改善 1 month ago
  みてるぞ 8c53efe707 #31 Densya-nai de no sagefu-bun 1 month ago
  みてるぞ f13c6d8260 #31 ニジカ足パタ対応 1 month ago
  みてるぞ d66153b87a #31 Xonzitu sagefu-bun 1 month ago
  みてるぞ 3f0825f835 #31 FPS を追記 1 month ago
  みてるぞ c83c99bc74 #31 本日作業分 1 month ago
  みてるぞ 9f705446a8 #31 ぼちぼち 1 month ago
  みてるぞ 11195e5125 #31 today sagefu-bun 1 month ago
  みてるぞ 57ae0992c1 #31 作りかけ 1 month ago
  みてるぞ a4ec55a4cd Merge branch 'main' into ai-migration 1 month ago
  みてるぞ dbfd6deb1e #30 AI 部分を排除 1 month ago
32 changed files with 918 additions and 804 deletions
Split View
  1. +1
    -1
      .gitignore
  2. +3
    -0
      .gitmodules
  3. BIN
      MusMus-BGM-097.mp3
  4. +0
    -0
      __init__.py
  5. +0
    -0
      assets/balloon.png
  6. +0
    -0
      assets/bg-evening.jpg
  7. +0
    -0
      assets/bg-grass.png
  8. +0
    -0
      assets/bg-night.jpg
  9. +0
    -0
      assets/bg.jpg
  10. BIN
      assets/bgm.mp3
  11. BIN
      assets/deerjika_relax_left.png
  12. +0
    -0
      assets/moon.png
  13. +0
    -0
      assets/noon.wav
  14. +0
    -0
      assets/oh.wav
  15. +0
    -0
      assets/snack_time.mp4
  16. +0
    -0
      assets/sun.png
  17. BIN
      bg.mp4
  18. BIN
      bg2.mp4
  19. +0
    -8
      common_const.py
  20. +0
    -2
      common_module.py
  21. +0
    -7
      connection.sample.py
  22. BIN
      goatoh.mp4
  23. BIN
      kusa.wav
  24. +911
    -451
      main.py
  25. +0
    -8
      mode.py
  26. BIN
      mumumumu.wav
  27. +2
    -0
      mypy.ini
  28. +1
    -0
      nizika_ai
  29. +0
    -44
      play_movie.py
  30. BIN
      snack_time.wav
  31. +0
    -280
      talk.py
  32. +0
    -3
      youtube.sample.py

+ 1
- 1
.gitignore View File

@@ -1,5 +1,5 @@
/connection.py
/__pycache__
__pycache__
/nizika_talking.wav
/youtube.py
/log.txt

+ 3
- 0
.gitmodules View File

@@ -0,0 +1,3 @@
[submodule "nizika_ai"]
path = nizika_ai
url = https://git.miteruzo.com/miteruzo/nizika_ai

BIN
MusMus-BGM-097.mp3 View File


+ 0
- 0
__init__.py View File


talking.png → assets/balloon.png View File


bg-evening.jpg → assets/bg-evening.jpg View File


bg-grass.png → assets/bg-grass.png View File


bg-night.jpg → assets/bg-night.jpg View File


bg.jpg → assets/bg.jpg View File


BIN
assets/bgm.mp3 View File


BIN
assets/deerjika_relax_left.png View File

Before After
Width: 47360  |  Height: 720  |  Size: 945 KiB

moon.png → assets/moon.png View File

Before After
Width: 1329  |  Height: 1329  |  Size: 641 KiB Width: 1329  |  Height: 1329  |  Size: 641 KiB

noon.wav → assets/noon.wav View File


oh.wav → assets/oh.wav View File


snack_time.mp4 → assets/snack_time.mp4 View File


sun.png → assets/sun.png View File


BIN
bg.mp4 View File


BIN
bg2.mp4 View File


+ 0
- 8
common_const.py View File

@@ -1,8 +0,0 @@
class CWindow:
WIDTH: int = 1024
HEIGHT: int = 768


class CMath:
PI: float = 3.14159265358979323846


+ 0
- 2
common_module.py View File

@@ -1,7 +1,5 @@
import unicodedata

from common_const import *


class CommonModule:
@staticmethod


+ 0
- 7
connection.sample.py View File

@@ -1,7 +0,0 @@
# 各変数に適切な値を設定し,ファイル名を connection.py として保存すること

# Organisation ID
OPENAI_ORGANISATION: str = 'org-XXXXXXXXXXXXXXXXXXXXXXXX'

# API Key
OPENAI_API_KEY: str = 'sk-XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX'

BIN
goatoh.mp4 View File


BIN
kusa.wav View File


+ 911
- 451
main.py
File diff suppressed because it is too large
View File


+ 0
- 8
mode.py View File

@@ -1,8 +0,0 @@
from enum import Enum, auto


class Mode (Enum):
NIZIKA = auto ()
GOATOH = auto ()
DOUBLE = auto ()


BIN
mumumumu.wav View File


+ 2
- 0
mypy.ini View File

@@ -0,0 +1,2 @@
[mypy]
disable_error_code = import-untyped

+ 1
- 0
nizika_ai

@@ -0,0 +1 @@
Subproject commit 9e136a7cb3baf97b384a07c4af31205d745e7200

+ 0
- 44
play_movie.py View File

@@ -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 ()

BIN
snack_time.wav View File


+ 0
- 280
talk.py View File

@@ -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 ''))

+ 0
- 3
youtube.sample.py View File

@@ -1,3 +0,0 @@
# 各変数に適切な値を設定し,ファイル名を youtube.py として保存すること

YOUTUBE_ID: str = 'XXXXXXXXXXX' # YouTube の配信 ID

Loading…
Cancel
Save