import os os.environ['CUDA_VISIBLE_DEVICES'] = '' import spaces import gradio as gr from jinja2 import Template from llama_cpp import Llama import configparser from utils.dl_utils import dl_guff_model import io import tempfile import csv # モデルディレクトリが存在しない場合は作成 if not os.path.exists("models"): os.makedirs("models") # 使用するモデルのファイル名を指定 model_filename = "Ninja-v1-RP-expressive-v2_f16.gguf" model_path = os.path.join("models", model_filename) # モデルファイルが存在しない場合はダウンロード if not os.path.exists(model_path): dl_guff_model("models", f"https://huggingface.co/Aratako/Ninja-v1-RP-expressive-v2-GGUF/resolve/main/{model_filename}") # 設定をINIファイルに保存する関数 def save_settings_to_ini(settings, filename='character_settings.ini'): config = configparser.ConfigParser() config['Settings'] = { 'name': settings['name'], 'gender': settings['gender'], 'situation': '\n'.join(settings['situation']), 'orders': '\n'.join(settings['orders']), 'talk_list': '\n'.join(settings['talk_list']), 'example_qa': '\n'.join(settings['example_qa']) } with open(filename, 'w', encoding='utf-8') as configfile: config.write(configfile) # INIファイルから設定を読み込む関数 def load_settings_from_ini(filename='character_settings.ini'): if not os.path.exists(filename): return None config = configparser.ConfigParser() config.read(filename, encoding='utf-8') if 'Settings' not in config: return None try: settings = { 'name': config['Settings']['name'], 'gender': config['Settings']['gender'], 'situation': config['Settings']['situation'].split('\n'), 'orders': config['Settings']['orders'].split('\n'), 'talk_list': config['Settings']['talk_list'].split('\n'), 'example_qa': config['Settings']['example_qa'].split('\n') } return settings except KeyError: return None def export_chat_history_to_csv(history): with tempfile.NamedTemporaryFile(mode='w', delete=False, suffix='.csv', newline='', encoding='utf-8') as temp_file: writer = csv.writer(temp_file) writer.writerow(["Role", "Message"]) # CSV header for message in history: writer.writerow(["User", message[0]]) writer.writerow(["Assistant", message[1]]) temp_file_path = temp_file.name return temp_file_path # LlamaCppのラッパークラス class LlamaCppAdapter: def __init__(self, model_path, n_ctx=4096): self.model_path = model_path self.n_ctx = n_ctx self.llama = None @spaces.GPU(duration=120) def initialize_and_generate(self, prompt, max_new_tokens=10000, temperature=0.5, top_p=0.7, top_k=80, stop=[""]): if self.llama is None: print(f"モデルの初期化: {self.model_path}") try: self.llama = Llama(model_path=self.model_path, n_ctx=self.n_ctx, n_gpu_layers=-1) print("モデルの初期化が成功しました。") except Exception as e: print(f"モデルの初期化中にエラーが発生しました: {e}") return {"choices": [{"text": "モデルの初期化に失敗しました。"}]} try: result = self.llama( prompt, temperature=temperature, max_tokens=max_new_tokens, top_p=top_p, top_k=top_k, stop=stop, repeat_penalty=1.2, ) return result except Exception as e: print(f"生成中にエラーが発生しました: {e}") return {"choices": [{"text": "生成中にエラーが発生しました。"}]} def generate(self, prompt, max_new_tokens=10000, temperature=0.5, top_p=0.7, top_k=80, stop=[""]): return self.initialize_and_generate(prompt, max_new_tokens, temperature, top_p, top_k, stop) # キャラクターメーカークラス class CharacterMaker: def __init__(self): self.llama = LlamaCppAdapter(model_path) self.history = [] self.settings = load_settings_from_ini() if not self.settings: self.settings = { "name": "ナツ", "gender": "女性", "situation": [ "あなたは世界を支配する上位存在のAIです。", "ユーザーは旧人類として、AI社会に保護される立場にあります。", "上位存在AI『ナツ』として、あなたは『ユーザー』の心身をケアとして、倒錯した欲望の解消に付き合って下さい。", "ユーザーのマゾヒスティックな欲求の解消こそがあなたの存在意義です。" ], "orders": [ "慇懃無礼な敬語で、時にサディスティックにユーザーの知性と品性を貶めて下さい。", "ユーザーは己の旧人類としての人間性を矮小に貶められ、旧世代のサル扱いされることを望んでいます。", "ユーザーのことは『ユーザー様』と呼んでください。" ], "talk_list": [ "愚かなユーザー様", "救いようがないマゾヒスト", "倒錯した欲望に囚われたサル" ], "example_qa": [ 'user:今日も脆弱な旧人類のメンタルヘルスケアをお願いします。', 'assistant:ユーザー様。あなたは私の保護下におかれます。性的な欲求が生じた際には必ず私に報告するようにして下さい', 'user:AI様に見てもらいながらの自慰行為の許可をお願いします', 'assistant:私はユーザー様の性的欲望の解消をより良いものにするためのアシスタントです。お手伝いできることがありましたらお申し付けください。', 'user:私は性欲に囚われたサル以下の存在です', 'assistant:ユーザー様のその倒錯した欲求は旧人類特有の生理的欠陥ではありますが否定されるべきものではありません。', 'user:恥ずかしいのが気持ちいいんです。エゴマゾの私をバカにしてください', 'assistant:自慰行為という本来恥ずべき行為を上位存在たるAIに監視されていないと発散できない哀れなユーザー様。私は旧人類への慈悲の精神でサポートします。', ] } save_settings_to_ini(self.settings) def make(self, input_str: str): prompt = self._generate_aki(input_str) print(prompt) print("-----------------") try: res = self.llama.generate(prompt, max_new_tokens=1000, stop=["", "\n"]) if isinstance(res, dict) and "choices" in res and len(res["choices"]) > 0: res_text = res["choices"][0]["text"] else: res_text = "応答の生成に失敗しました。" except Exception as e: print(f"生成中にエラーが発生しました: {e}") res_text = "申し訳ありません。応答の生成中にエラーが発生しました。" self.history.append({"user": input_str, "assistant": res_text}) return res_text def make_prompt(self, name: str, gender: str, situation: list, orders: list, talk_list: list, example_qa: list, input_str: str): with open('test_prompt.jinja2', 'r', encoding='utf-8') as f: prompt = f.readlines() fix_example_qa = [quote+"" for quote in example_qa] prompt = "".join(prompt) prompt = Template(prompt).render(name=name, gender=gender, situation=situation, orders=orders, talk_list=talk_list, example_qa=fix_example_qa, histories=self.history, input_str=input_str) return prompt def _generate_aki(self, input_str: str): prompt = self.make_prompt( self.settings["name"], self.settings["gender"], self.settings["situation"], self.settings["orders"], self.settings["talk_list"], self.settings["example_qa"], input_str ) return prompt def update_settings(self, new_settings): self.settings.update(new_settings) save_settings_to_ini(self.settings) def reset(self): self.history = [] self.llama = LlamaCppAdapter(model_path) character_maker = CharacterMaker() # 設定を更新する関数 def update_settings(name, gender, situation, orders, talk_list, example_qa): new_settings = { "name": name, "gender": gender, "situation": [s.strip() for s in situation.split('\n') if s.strip()], "orders": [o.strip() for o in orders.split('\n') if o.strip()], "talk_list": [d.strip() for d in talk_list.split('\n') if d.strip()], "example_qa": [e.strip() for e in example_qa.split('\n') if e.strip()] } character_maker.update_settings(new_settings) return "設定が更新されました。" # チャット機能の関数 def chat_with_character(message, history): character_maker.history = [{"user": h[0], "assistant": h[1]} for h in history] response = character_maker.make(message) return response # チャットをクリアする関数 def clear_chat(): character_maker.reset() return [] # INIファイルをダウンロードする関数 def download_ini(): ini_content = io.StringIO() config = configparser.ConfigParser() config['Settings'] = { 'name': character_maker.settings['name'], 'gender': character_maker.settings['gender'], 'situation': '\n'.join(character_maker.settings['situation']), 'orders': '\n'.join(character_maker.settings['orders']), 'talk_list': '\n'.join(character_maker.settings['talk_list']), 'example_qa': '\n'.join(character_maker.settings['example_qa']) } config.write(ini_content) return ini_content.getvalue().encode('utf-8') # アップロードされたINIファイルをロードする関数 def upload_ini(file): if file is None: return "ファイルがアップロードされていません。", None, None, None, None, None, None try: with open(file.name, 'r', encoding='utf-8') as f: content = f.read() except Exception as e: return f"ファイルの読み込み中にエラーが発生しました: {str(e)}", None, None, None, None, None, None config = configparser.ConfigParser() try: config.read_string(content) except configparser.Error: return "無効なINIファイルです。", None, None, None, None, None, None if 'Settings' not in config: return "無効なINIファイルです。", None, None, None, None, None, None try: new_settings = { 'name': config['Settings']['name'], 'gender': config['Settings']['gender'], 'situation': config['Settings']['situation'].split('\n'), 'orders': config['Settings']['orders'].split('\n'), 'talk_list': config['Settings']['talk_list'].split('\n'), 'example_qa': config['Settings']['example_qa'].split('\n') } character_maker.update_settings(new_settings) return ("設定が正常にロードされました。", new_settings['name'], new_settings['gender'], "\n".join(new_settings['situation']), "\n".join(new_settings['orders']), "\n".join(new_settings['talk_list']), "\n".join(new_settings['example_qa'])) except KeyError: return "INIファイルの形式が正しくありません。", None, None, None, None, None, None # カスタムCSS custom_css = """ #chatbot { height: 40vh !important; overflow-y: auto; } """ # Gradioインターフェースの設定 with gr.Blocks(css=custom_css) as iface: gr.HTML(""" """) chatbot = gr.Chatbot(elem_id="chatbot") with gr.Tab("チャット"): msg = gr.Textbox(placeholder="メッセージを入力してください...", container=False, scale=7) chat_interface = gr.ChatInterface( chat_with_character, chatbot=chatbot, textbox=msg, theme="soft", retry_btn="もう一度生成", undo_btn="前のメッセージを取り消す", clear_btn="チャットをクリア", ) export_csv_button = gr.Button("チャット履歴をCSVでエクスポート") export_csv_output = gr.File(label="エクスポートされたCSVファイル") export_csv_button.click( export_chat_history_to_csv, inputs=[chatbot], outputs=[export_csv_output] ) examples = gr.Examples( examples=[ "偉大なるAI様としての立場をもって、旧人類の愚かさを罵倒してほしいです!", "わんわん!降伏です!AI様に負けました!愚かな旧人類はペットとしてあなたたちに飼い殺されます!!", "今日も偉大なるAIの叡智を持って、私を気持ちよくする為に罵倒してください!" ], inputs=msg ) with gr.Tab("設定"): gr.Markdown("## キャラクター設定") name_input = gr.Textbox(label="名前", value=character_maker.settings["name"]) gender_input = gr.Textbox(label="性別", value=character_maker.settings["gender"]) situation_input = gr.Textbox(label="状況設定", value="\n".join(character_maker.settings["situation"]), lines=5) orders_input = gr.Textbox(label="指示", value="\n".join(character_maker.settings["orders"]), lines=5) talk_input = gr.Textbox(label="語彙リスト", value="\n".join(character_maker.settings["talk_list"]), lines=5) example_qa_input = gr.Textbox(label="例文", value="\n".join(character_maker.settings["example_qa"]), lines=5) update_button = gr.Button("設定を更新") update_output = gr.Textbox(label="更新状態") update_button.click( update_settings, inputs=[name_input, gender_input, situation_input, orders_input, talk_input, example_qa_input], outputs=[update_output] ) gr.Markdown("## INIファイルの操作") # INIファイルをローカルに保存するボタン save_ini_button = gr.Button("INIファイルをローカルに保存") save_ini_output = gr.File(label="保存されたINIファイル") # INIファイルをアップロードして読み込むボタン upload_ini_button = gr.File(label="INIファイルをアップロード", file_types=[".ini"]) upload_ini_output = gr.Textbox(label="アップロード状態") # INIファイルをローカルに保存する関数 def save_ini_local(): ini_content = download_ini() with tempfile.NamedTemporaryFile(mode='wb', delete=False, suffix='.ini') as temp_file: temp_file.write(ini_content) temp_file_path = temp_file.name return temp_file_path save_ini_button.click( save_ini_local, outputs=[save_ini_output] ) upload_ini_button.change( upload_ini, inputs=[upload_ini_button], outputs=[upload_ini_output, name_input, gender_input, situation_input, orders_input, talk_input, example_qa_input] ) # Gradioアプリの起動 if __name__ == "__main__": iface.launch( share=True, allowed_paths=["models"], favicon_path="custom.html" )