2022-12-21 11:27:31 -05:00
import re
2023-01-21 22:02:46 -05:00
import gc
2023-01-05 23:33:21 -05:00
import time
import glob
2022-12-21 11:27:31 -05:00
import torch
2023-01-06 17:56:44 -05:00
import argparse
2023-01-15 13:23:41 -05:00
import json
2023-01-28 17:16:37 -05:00
import io
2023-01-28 18:18:23 -05:00
import base64
2023-01-26 22:40:39 -05:00
import sys
2023-01-21 22:02:46 -05:00
from sys import exit
2023-01-07 14:33:43 -05:00
from pathlib import Path
2023-01-28 17:16:37 -05:00
from PIL import Image
2023-01-27 10:01:11 -05:00
import copy
2022-12-21 11:27:31 -05:00
import gradio as gr
2023-01-14 22:39:51 -05:00
import warnings
2023-01-19 10:20:57 -05:00
from tqdm import tqdm
2023-01-21 22:02:46 -05:00
import transformers
from transformers import AutoTokenizer , AutoModelForCausalLM
from modules . html_generator import *
from modules . ui import *
2023-01-25 08:17:55 -05:00
from modules . stopping_criteria import _SentinelTokenStoppingCriteria
2022-12-21 11:27:31 -05:00
2023-01-15 13:23:41 -05:00
transformers . logging . set_verbosity_error ( )
2023-01-06 17:56:44 -05:00
parser = argparse . ArgumentParser ( )
2023-01-06 18:22:26 -05:00
parser . add_argument ( ' --model ' , type = str , help = ' Name of the model to load by default. ' )
2023-01-16 08:10:09 -05:00
parser . add_argument ( ' --notebook ' , action = ' store_true ' , help = ' Launch the web UI in notebook mode, where the output is written to the same text box as the input. ' )
parser . add_argument ( ' --chat ' , action = ' store_true ' , help = ' Launch the web UI in chat mode. ' )
2023-01-25 17:39:36 -05:00
parser . add_argument ( ' --cai-chat ' , action = ' store_true ' , help = ' Launch the web UI in chat mode with a style similar to Character.AI \' s. If the file img_bot.png or img_bot.jpg exists in the same folder as server.py, this image will be used as the bot \' s profile picture. Similarly, img_me.png or img_me.jpg will be used as your profile picture. ' )
2023-01-09 08:58:46 -05:00
parser . add_argument ( ' --cpu ' , action = ' store_true ' , help = ' Use the CPU to generate text. ' )
2023-01-10 21:16:33 -05:00
parser . add_argument ( ' --load-in-8bit ' , action = ' store_true ' , help = ' Load the model with 8-bit precision. ' )
2023-01-19 09:09:24 -05:00
parser . add_argument ( ' --auto-devices ' , action = ' store_true ' , help = ' Automatically split the model across the available GPU(s) and CPU. ' )
parser . add_argument ( ' --disk ' , action = ' store_true ' , help = ' If the model is too large for your GPU(s) and CPU combined, send the remaining layers to the disk. ' )
2023-01-21 00:48:06 -05:00
parser . add_argument ( ' --disk-cache-dir ' , type = str , help = ' Directory to save the disk cache to. Defaults to " cache/ " . ' )
2023-01-20 22:33:41 -05:00
parser . add_argument ( ' --gpu-memory ' , type = int , help = ' Maximum GPU memory in GiB to allocate. This is useful if you get out of memory errors while trying to generate text. Must be an integer number. ' )
2023-01-21 01:05:55 -05:00
parser . add_argument ( ' --cpu-memory ' , type = int , help = ' Maximum CPU memory in GiB to allocate for offloaded weights. Must be an integer number. Defaults to 99. ' )
2023-01-22 14:19:11 -05:00
parser . add_argument ( ' --no-stream ' , action = ' store_true ' , help = ' Don \' t stream the text output in real time. This improves the text generation performance. ' )
2023-01-16 14:35:45 -05:00
parser . add_argument ( ' --settings ' , type = str , help = ' Load the default interface settings from this json file. See settings-template.json for an example. ' )
2023-01-26 22:40:39 -05:00
parser . add_argument ( ' --extensions ' , type = str , help = ' The list of extensions to load. If you want to load more than one extension, write the names separated by commas and between quotation marks, " like,this " . ' )
2023-01-20 21:45:16 -05:00
parser . add_argument ( ' --listen ' , action = ' store_true ' , help = ' Make the web UI reachable from your local network. ' )
2023-01-29 00:54:36 -05:00
parser . add_argument ( ' --listen-port ' , type = int , help = ' The listening port that the server will use. ' )
2023-01-19 15:31:29 -05:00
parser . add_argument ( ' --share ' , action = ' store_true ' , help = ' Create a public URL. This is useful for running the web UI on Google Colab or similar. ' )
2023-01-26 00:12:53 -05:00
parser . add_argument ( ' --verbose ' , action = ' store_true ' , help = ' Print the prompts to the terminal. ' )
2023-01-06 17:56:44 -05:00
args = parser . parse_args ( )
2023-01-14 22:39:51 -05:00
2023-01-22 14:19:11 -05:00
if ( args . chat or args . cai_chat ) and not args . no_stream :
2023-01-25 12:37:41 -05:00
print ( " Warning: chat mode currently becomes somewhat slower with text streaming on. \n Consider starting the web UI with the --no-stream option. \n " )
2023-01-22 14:19:11 -05:00
2023-01-15 13:23:41 -05:00
settings = {
' max_new_tokens ' : 200 ,
' max_new_tokens_min ' : 1 ,
' max_new_tokens_max ' : 2000 ,
' preset ' : ' NovelAI-Sphinx Moth ' ,
' name1 ' : ' Person 1 ' ,
' name2 ' : ' Person 2 ' ,
' context ' : ' This is a conversation between two people. ' ,
' prompt ' : ' Common sense questions and answers \n \n Question: \n Factual answer: ' ,
' prompt_gpt4chan ' : ' ----- \n --- 865467536 \n Input text \n --- 865467537 \n ' ,
' stop_at_newline ' : True ,
2023-01-22 15:17:35 -05:00
' history_size ' : 0 ,
2023-01-20 15:03:09 -05:00
' history_size_min ' : 0 ,
' history_size_max ' : 64 ,
2023-01-19 16:58:45 -05:00
' preset_pygmalion ' : ' Pygmalion ' ,
' name1_pygmalion ' : ' You ' ,
' name2_pygmalion ' : ' Kawaii ' ,
2023-01-21 22:49:59 -05:00
' context_pygmalion ' : " Kawaii ' s persona: Kawaii is a cheerful person who loves to make others smile. She is an optimist who loves to spread happiness and positivity wherever she goes. \n <START> " ,
2023-01-19 14:46:46 -05:00
' stop_at_newline_pygmalion ' : False ,
2023-01-15 13:23:41 -05:00
}
2023-01-16 14:35:45 -05:00
if args . settings is not None and Path ( args . settings ) . exists ( ) :
with open ( Path ( args . settings ) , ' r ' ) as f :
2023-01-15 13:23:41 -05:00
new_settings = json . load ( f )
2023-01-16 14:35:45 -05:00
for item in new_settings :
2023-01-28 21:21:40 -05:00
settings [ item ] = new_settings [ item ]
2023-01-14 22:39:51 -05:00
2022-12-21 11:27:31 -05:00
def load_model ( model_name ) :
2023-01-05 23:41:52 -05:00
print ( f " Loading { model_name } ... " )
2022-12-21 11:27:31 -05:00
t0 = time . time ( )
2023-01-05 23:41:52 -05:00
2023-01-10 21:16:33 -05:00
# Default settings
2023-01-20 21:45:16 -05:00
if not ( args . cpu or args . load_in_8bit or args . auto_devices or args . disk or args . gpu_memory is not None ) :
2023-01-10 21:16:33 -05:00
if Path ( f " torch-dumps/ { model_name } .pt " ) . exists ( ) :
print ( " Loading in .pt format... " )
model = torch . load ( Path ( f " torch-dumps/ { model_name } .pt " ) )
elif model_name . lower ( ) . startswith ( ( ' gpt-neo ' , ' opt- ' , ' galactica ' ) ) and any ( size in model_name . lower ( ) for size in ( ' 13b ' , ' 20b ' , ' 30b ' ) ) :
model = AutoModelForCausalLM . from_pretrained ( Path ( f " models/ { model_name } " ) , device_map = ' auto ' , load_in_8bit = True )
else :
model = AutoModelForCausalLM . from_pretrained ( Path ( f " models/ { model_name } " ) , low_cpu_mem_usage = True , torch_dtype = torch . float16 ) . cuda ( )
# Custom
2023-01-06 00:54:33 -05:00
else :
2023-01-10 21:16:33 -05:00
settings = [ " low_cpu_mem_usage=True " ]
2023-01-10 21:39:50 -05:00
command = " AutoModelForCausalLM.from_pretrained "
2023-01-10 21:16:33 -05:00
2023-01-09 14:28:04 -05:00
if args . cpu :
2023-01-10 21:16:33 -05:00
settings . append ( " torch_dtype=torch.float32 " )
2023-01-09 14:28:04 -05:00
else :
2023-01-15 21:01:51 -05:00
settings . append ( " device_map= ' auto ' " )
2023-01-20 21:45:16 -05:00
if args . gpu_memory is not None :
2023-01-20 22:33:41 -05:00
if args . cpu_memory is not None :
settings . append ( f " max_memory= {{ 0: ' { args . gpu_memory } GiB ' , ' cpu ' : ' { args . cpu_memory } GiB ' }} " )
2023-01-20 17:05:43 -05:00
else :
2023-01-20 22:25:34 -05:00
settings . append ( f " max_memory= {{ 0: ' { args . gpu_memory } GiB ' , ' cpu ' : ' 99GiB ' }} " )
2023-01-19 09:09:24 -05:00
if args . disk :
2023-01-20 17:05:43 -05:00
if args . disk_cache_dir is not None :
2023-01-21 13:04:13 -05:00
settings . append ( f " offload_folder= ' { args . disk_cache_dir } ' " )
2023-01-20 17:05:43 -05:00
else :
settings . append ( " offload_folder= ' cache ' " )
2023-01-15 21:01:51 -05:00
if args . load_in_8bit :
2023-01-10 21:16:33 -05:00
settings . append ( " load_in_8bit=True " )
else :
settings . append ( " torch_dtype=torch.float16 " )
2023-01-16 14:35:45 -05:00
settings = ' , ' . join ( set ( settings ) )
2023-01-15 21:01:51 -05:00
command = f " { command } (Path(f ' models/ { model_name } ' ), { settings } ) "
2023-01-10 21:16:33 -05:00
model = eval ( command )
2022-12-21 11:27:31 -05:00
2023-01-06 00:54:33 -05:00
# Loading the tokenizer
2023-01-10 23:10:11 -05:00
if model_name . lower ( ) . startswith ( ( ' gpt4chan ' , ' gpt-4chan ' , ' 4chan ' ) ) and Path ( f " models/gpt-j-6B/ " ) . exists ( ) :
2023-01-07 14:33:43 -05:00
tokenizer = AutoTokenizer . from_pretrained ( Path ( " models/gpt-j-6B/ " ) )
2022-12-21 11:27:31 -05:00
else :
2023-01-07 14:33:43 -05:00
tokenizer = AutoTokenizer . from_pretrained ( Path ( f " models/ { model_name } / " ) )
2023-01-16 11:43:23 -05:00
tokenizer . truncation_side = ' left '
2022-12-21 11:27:31 -05:00
2023-01-06 00:06:59 -05:00
print ( f " Loaded the model in { ( time . time ( ) - t0 ) : .2f } seconds. " )
2022-12-21 11:27:31 -05:00
return model , tokenizer
2023-01-06 00:26:33 -05:00
# Removes empty replies from gpt4chan outputs
2022-12-21 11:27:31 -05:00
def fix_gpt4chan ( s ) :
for i in range ( 10 ) :
s = re . sub ( " --- [0-9]* \n >>[0-9]* \n --- " , " --- " , s )
s = re . sub ( " --- [0-9]* \n * \n --- " , " --- " , s )
s = re . sub ( " --- [0-9]* \n \n \n --- " , " --- " , s )
return s
2023-01-16 14:35:45 -05:00
# Fix the LaTeX equations in galactica
2023-01-06 23:56:21 -05:00
def fix_galactica ( s ) :
s = s . replace ( r ' \ [ ' , r ' $ ' )
s = s . replace ( r ' \ ] ' , r ' $ ' )
2023-01-07 10:13:09 -05:00
s = s . replace ( r ' \ ( ' , r ' $ ' )
s = s . replace ( r ' \ ) ' , r ' $ ' )
s = s . replace ( r ' $$ ' , r ' $ ' )
2023-01-06 23:56:21 -05:00
return s
2023-01-25 08:17:55 -05:00
def encode ( prompt , tokens_to_generate = 0 , add_special_tokens = True ) :
2023-01-23 11:36:01 -05:00
if args . cpu :
2023-01-25 08:17:55 -05:00
input_ids = tokenizer . encode ( str ( prompt ) , return_tensors = ' pt ' , truncation = True , max_length = 2048 - tokens_to_generate , add_special_tokens = add_special_tokens )
2023-01-23 11:36:01 -05:00
else :
2023-01-17 18:16:23 -05:00
torch . cuda . empty_cache ( )
2023-01-25 08:17:55 -05:00
input_ids = tokenizer . encode ( str ( prompt ) , return_tensors = ' pt ' , truncation = True , max_length = 2048 - tokens_to_generate , add_special_tokens = add_special_tokens ) . cuda ( )
2023-01-17 18:16:23 -05:00
return input_ids
2023-01-19 08:43:05 -05:00
def decode ( output_ids ) :
reply = tokenizer . decode ( output_ids , skip_special_tokens = True )
reply = reply . replace ( r ' <|endoftext|> ' , ' ' )
return reply
def formatted_outputs ( reply , model_name ) :
2023-01-19 12:57:01 -05:00
if not ( args . chat or args . cai_chat ) :
if model_name . lower ( ) . startswith ( ' galactica ' ) :
reply = fix_galactica ( reply )
return reply , reply , generate_basic_html ( reply )
2023-01-21 20:13:01 -05:00
elif model_name . lower ( ) . startswith ( ( ' gpt4chan ' , ' gpt-4chan ' , ' 4chan ' ) ) :
2023-01-19 12:57:01 -05:00
reply = fix_gpt4chan ( reply )
return reply , ' Only applicable for GALACTICA models. ' , generate_4chan_html ( reply )
else :
return reply , ' Only applicable for GALACTICA models. ' , generate_basic_html ( reply )
2023-01-19 08:43:05 -05:00
else :
2023-01-19 12:57:01 -05:00
return reply
2023-01-19 08:43:05 -05:00
2023-01-25 08:17:55 -05:00
def generate_reply ( question , tokens , inference_settings , selected_model , eos_token = None , stopping_string = None ) :
2023-01-06 00:06:59 -05:00
global model , tokenizer , model_name , loaded_preset , preset
2022-12-21 11:27:31 -05:00
2023-01-26 22:40:39 -05:00
original_question = question
if not ( args . chat or args . cai_chat ) :
question = apply_extensions ( question , " input " )
2023-01-26 00:12:53 -05:00
if args . verbose :
print ( f " \n \n { question } \n -------------------- \n " )
2022-12-21 11:27:31 -05:00
if selected_model != model_name :
model_name = selected_model
2023-01-20 21:45:16 -05:00
model = tokenizer = None
2023-01-09 08:58:46 -05:00
if not args . cpu :
2023-01-19 10:01:58 -05:00
gc . collect ( )
2023-01-09 08:58:46 -05:00
torch . cuda . empty_cache ( )
2022-12-21 11:27:31 -05:00
model , tokenizer = load_model ( model_name )
2023-01-06 00:06:59 -05:00
if inference_settings != loaded_preset :
2023-01-07 14:33:43 -05:00
with open ( Path ( f ' presets/ { inference_settings } .txt ' ) , ' r ' ) as infile :
2023-01-05 23:33:21 -05:00
preset = infile . read ( )
2023-01-06 00:06:59 -05:00
loaded_preset = inference_settings
2022-12-21 11:27:31 -05:00
2023-01-18 22:41:57 -05:00
cuda = " " if args . cpu else " .cuda() "
2023-01-25 20:27:04 -05:00
n = tokenizer . eos_token_id if eos_token is None else tokenizer . encode ( eos_token , return_tensors = ' pt ' ) [ 0 ] [ - 1 ]
2023-01-19 22:45:02 -05:00
input_ids = encode ( question , tokens )
2023-01-25 08:17:55 -05:00
if stopping_string is not None :
2023-01-26 11:45:19 -05:00
# The stopping_criteria code below was copied from
# https://github.com/PygmalionAI/gradio-ui/blob/master/src/model.py
2023-01-25 08:17:55 -05:00
t = encode ( stopping_string , 0 , add_special_tokens = False )
stopping_criteria_list = transformers . StoppingCriteriaList ( [
_SentinelTokenStoppingCriteria (
2023-01-25 17:37:44 -05:00
sentinel_token_ids = t ,
starting_idx = len ( input_ids [ 0 ] )
)
2023-01-25 08:17:55 -05:00
] )
else :
stopping_criteria_list = None
2023-01-19 08:43:05 -05:00
# Generate the entire reply at once
if args . no_stream :
2023-01-22 18:07:19 -05:00
t0 = time . time ( )
2023-01-25 08:17:55 -05:00
output = eval ( f " model.generate(input_ids, eos_token_id= { n } , stopping_criteria=stopping_criteria_list, { preset } ) { cuda } " )
2023-01-19 08:43:05 -05:00
reply = decode ( output [ 0 ] )
2023-01-22 18:07:19 -05:00
t1 = time . time ( )
print ( f " Output generated in { ( t1 - t0 ) : .2f } seconds ( { ( len ( output [ 0 ] ) - len ( input_ids [ 0 ] ) ) / ( t1 - t0 ) : .2f } it/s) " )
2023-01-26 22:40:39 -05:00
if not ( args . chat or args . cai_chat ) :
reply = original_question + apply_extensions ( reply [ len ( question ) : ] , " output " )
2023-01-19 08:43:05 -05:00
yield formatted_outputs ( reply , model_name )
# Generate the reply 1 token at a time
else :
2023-01-26 22:40:39 -05:00
yield formatted_outputs ( original_question , model_name )
2023-01-25 08:38:26 -05:00
preset = preset . replace ( ' max_new_tokens=tokens ' , ' max_new_tokens=8 ' )
for i in tqdm ( range ( tokens / / 8 + 1 ) ) :
2023-01-25 08:17:55 -05:00
output = eval ( f " model.generate(input_ids, eos_token_id= { n } , stopping_criteria=stopping_criteria_list, { preset } ) { cuda } " )
2023-01-19 08:43:05 -05:00
reply = decode ( output [ 0 ] )
2023-01-26 22:40:39 -05:00
if not ( args . chat or args . cai_chat ) :
reply = original_question + apply_extensions ( reply [ len ( question ) : ] , " output " )
2023-01-19 08:43:05 -05:00
yield formatted_outputs ( reply , model_name )
2023-01-18 21:56:42 -05:00
input_ids = output
2023-01-25 20:27:04 -05:00
if output [ 0 ] [ - 1 ] == n :
break
2023-01-18 19:37:21 -05:00
2023-01-26 22:40:39 -05:00
def apply_extensions ( text , typ ) :
global available_extensions , extension_state
for ext in sorted ( extension_state , key = lambda x : extension_state [ x ] [ 1 ] ) :
if extension_state [ ext ] [ 0 ] == True :
ext_string = f " extensions. { ext } .script "
if typ == " input " :
text = eval ( f " { ext_string } .input_modifier(text) " )
else :
text = eval ( f " { ext_string } .output_modifier(text) " )
return text
2023-01-21 22:49:59 -05:00
def get_available_models ( ) :
return sorted ( set ( [ item . replace ( ' .pt ' , ' ' ) for item in map ( lambda x : str ( x . name ) , list ( Path ( ' models/ ' ) . glob ( ' * ' ) ) + list ( Path ( ' torch-dumps/ ' ) . glob ( ' * ' ) ) ) if not item . endswith ( ' .txt ' ) ] ) , key = str . lower )
def get_available_presets ( ) :
return sorted ( set ( map ( lambda x : ' . ' . join ( str ( x . name ) . split ( ' . ' ) [ : - 1 ] ) , Path ( ' presets ' ) . glob ( ' *.txt ' ) ) ) , key = str . lower )
def get_available_characters ( ) :
return [ " None " ] + sorted ( set ( map ( lambda x : ' . ' . join ( str ( x . name ) . split ( ' . ' ) [ : - 1 ] ) , Path ( ' characters ' ) . glob ( ' *.json ' ) ) ) , key = str . lower )
2023-01-26 22:40:39 -05:00
def get_available_extensions ( ) :
return sorted ( set ( map ( lambda x : x . parts [ 1 ] , Path ( ' extensions ' ) . glob ( ' */script.py ' ) ) ) , key = str . lower )
2023-01-21 22:49:59 -05:00
available_models = get_available_models ( )
available_presets = get_available_presets ( )
available_characters = get_available_characters ( )
2023-01-26 22:40:39 -05:00
available_extensions = get_available_extensions ( )
extension_state = { }
if args . extensions is not None :
for i , ext in enumerate ( args . extensions . split ( ' , ' ) ) :
if ext in available_extensions :
2023-01-27 08:53:05 -05:00
print ( f ' Loading the extension " { ext } " ... ' , end = ' ' )
ext_string = f " extensions. { ext } .script "
exec ( f " import { ext_string } " )
2023-01-26 22:40:39 -05:00
extension_state [ ext ] = [ True , i ]
2023-01-27 08:53:05 -05:00
print ( f ' Ok. ' )
2023-01-21 22:49:59 -05:00
2023-01-06 17:56:44 -05:00
# Choosing the default model
if args . model is not None :
model_name = args . model
else :
2023-01-06 20:05:37 -05:00
if len ( available_models ) == 0 :
2023-01-06 17:56:44 -05:00
print ( " No models are available! Please download at least one. " )
exit ( 0 )
elif len ( available_models ) == 1 :
i = 0
else :
print ( " The following models are available: \n " )
for i , model in enumerate ( available_models ) :
print ( f " { i + 1 } . { model } " )
print ( f " \n Which one do you want to load? 1- { len ( available_models ) } \n " )
i = int ( input ( ) ) - 1
2023-01-09 10:56:54 -05:00
print ( )
2023-01-06 17:56:44 -05:00
model_name = available_models [ i ]
2022-12-21 11:27:31 -05:00
model , tokenizer = load_model ( model_name )
2023-01-21 22:49:59 -05:00
loaded_preset = None
2023-01-06 17:56:44 -05:00
2023-01-08 18:10:31 -05:00
# UI settings
2023-01-21 22:49:59 -05:00
default_text = settings [ ' prompt_gpt4chan ' ] if model_name . lower ( ) . startswith ( ( ' gpt4chan ' , ' gpt-4chan ' , ' 4chan ' ) ) else settings [ ' prompt ' ]
2023-01-15 13:23:41 -05:00
description = f " \n \n # Text generation lab \n Generate text using Large Language Models. \n "
2023-01-21 22:02:46 -05:00
css = " .my-4 { margin-top: 0} .py-6 { padding-top: 2.5rem} #refresh-button { flex: none; margin: 0; padding: 0; min-width: 50px; border: none; box-shadow: none; border-radius: 0} #download-label, #upload-label { min-height: 0} "
2023-01-21 22:49:59 -05:00
2023-01-18 20:44:47 -05:00
if args . chat or args . cai_chat :
2023-01-26 22:40:39 -05:00
history = { ' internal ' : [ ] , ' visible ' : [ ] }
2023-01-19 14:46:46 -05:00
character = None
2023-01-07 20:52:46 -05:00
2023-01-14 22:39:51 -05:00
# This gets the new line characters right.
2023-01-18 17:06:50 -05:00
def clean_chat_message ( text ) :
2023-01-14 21:50:34 -05:00
text = text . replace ( ' \n ' , ' \n \n ' )
text = re . sub ( r " \ n { 3,} " , " \n \n " , text )
text = text . strip ( )
2023-01-14 22:39:51 -05:00
return text
2023-01-20 15:03:09 -05:00
def generate_chat_prompt ( text , tokens , name1 , name2 , context , history_size ) :
2023-01-18 17:06:50 -05:00
text = clean_chat_message ( text )
2023-01-14 21:50:34 -05:00
2023-01-19 23:54:38 -05:00
rows = [ f " { context . strip ( ) } \n " ]
2023-01-26 22:40:39 -05:00
i = len ( history [ ' internal ' ] ) - 1
2023-01-20 15:03:09 -05:00
count = 0
2023-01-17 18:16:23 -05:00
while i > = 0 and len ( encode ( ' ' . join ( rows ) , tokens ) [ 0 ] ) < 2048 - tokens :
2023-01-26 22:40:39 -05:00
rows . insert ( 1 , f " { name2 } : { history [ ' internal ' ] [ i ] [ 1 ] . strip ( ) } \n " )
2023-01-20 15:03:09 -05:00
count + = 1
2023-01-26 22:40:39 -05:00
if not ( history [ ' internal ' ] [ i ] [ 0 ] == ' <|BEGIN-VISIBLE-CHAT|> ' ) :
rows . insert ( 1 , f " { name1 } : { history [ ' internal ' ] [ i ] [ 0 ] . strip ( ) } \n " )
2023-01-20 15:03:09 -05:00
count + = 1
2023-01-17 18:16:23 -05:00
i - = 1
2023-01-20 15:03:09 -05:00
if history_size != 0 and count > = history_size :
break
2023-01-17 18:16:23 -05:00
rows . append ( f " { name1 } : { text } \n " )
rows . append ( f " { name2 } : " )
while len ( rows ) > 3 and len ( encode ( ' ' . join ( rows ) , tokens ) [ 0 ] ) > = 2048 - tokens :
rows . pop ( 1 )
rows . pop ( 1 )
question = ' ' . join ( rows )
2023-01-18 17:06:50 -05:00
return question
2023-01-07 20:52:46 -05:00
2023-01-20 15:03:09 -05:00
def chatbot_wrapper ( text , tokens , inference_settings , selected_model , name1 , name2 , context , check , history_size ) :
2023-01-26 22:40:39 -05:00
original_text = text
text = apply_extensions ( text , " input " )
2023-01-20 15:03:09 -05:00
question = generate_chat_prompt ( text , tokens , name1 , name2 , context , history_size )
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] . append ( [ ' ' , ' ' ] )
history [ ' visible ' ] . append ( [ ' ' , ' ' ] )
2023-01-18 17:06:50 -05:00
eos_token = ' \n ' if check else None
2023-01-25 08:17:55 -05:00
for reply in generate_reply ( question , tokens , inference_settings , selected_model , eos_token = eos_token , stopping_string = f " \n { name1 } : " ) :
2023-01-18 19:51:18 -05:00
next_character_found = False
2023-01-18 17:06:50 -05:00
2023-01-19 17:59:34 -05:00
previous_idx = [ m . start ( ) for m in re . finditer ( f " (^| \n ) { name2 } : " , question ) ]
2023-01-19 12:57:01 -05:00
idx = [ m . start ( ) for m in re . finditer ( f " (^| \n ) { name2 } : " , reply ) ]
idx = idx [ len ( previous_idx ) - 1 ]
2023-01-19 17:59:34 -05:00
2023-01-19 12:57:01 -05:00
reply = reply [ idx + len ( f " \n { name2 } : " ) : ]
2023-01-18 17:06:50 -05:00
if check :
2023-01-19 12:57:01 -05:00
reply = reply . split ( ' \n ' ) [ 0 ] . strip ( )
2023-01-18 17:06:50 -05:00
else :
idx = reply . find ( f " \n { name1 } : " )
if idx != - 1 :
reply = reply [ : idx ]
2023-01-18 19:51:18 -05:00
next_character_found = True
2023-01-18 17:06:50 -05:00
reply = clean_chat_message ( reply )
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] [ - 1 ] = [ text , reply ]
history [ ' visible ' ] [ - 1 ] = [ original_text , apply_extensions ( reply , " output " ) ]
2023-01-19 08:43:05 -05:00
if next_character_found :
break
2023-01-18 17:06:50 -05:00
# Prevent the chat log from flashing if something like "\nYo" is generated just
# before "\nYou:" is completed
tmp = f " \n { name1 } : "
2023-01-18 19:51:18 -05:00
next_character_substring_found = False
2023-01-19 08:43:05 -05:00
for j in range ( 1 , len ( tmp ) ) :
2023-01-18 17:06:50 -05:00
if reply [ - j : ] == tmp [ : j ] :
2023-01-18 19:51:18 -05:00
next_character_substring_found = True
2023-01-18 17:06:50 -05:00
2023-01-18 19:51:18 -05:00
if not next_character_substring_found :
2023-01-26 22:40:39 -05:00
yield history [ ' visible ' ]
2023-01-07 20:52:46 -05:00
2023-01-26 22:40:39 -05:00
yield history [ ' visible ' ]
2023-01-18 19:51:18 -05:00
2023-01-20 15:03:09 -05:00
def cai_chatbot_wrapper ( text , tokens , inference_settings , selected_model , name1 , name2 , context , check , history_size ) :
2023-01-26 22:40:39 -05:00
for _history in chatbot_wrapper ( text , tokens , inference_settings , selected_model , name1 , name2 , context , check , history_size ) :
yield generate_chat_html ( _history , name1 , name2 , character )
2023-01-15 10:20:04 -05:00
2023-01-22 00:19:58 -05:00
def regenerate_wrapper ( text , tokens , inference_settings , selected_model , name1 , name2 , context , check , history_size ) :
2023-01-27 09:14:19 -05:00
last = history [ ' visible ' ] . pop ( )
history [ ' internal ' ] . pop ( )
2023-01-22 00:19:58 -05:00
text = last [ 0 ]
if args . cai_chat :
for i in cai_chatbot_wrapper ( text , tokens , inference_settings , selected_model , name1 , name2 , context , check , history_size ) :
yield i
else :
for i in chatbot_wrapper ( text , tokens , inference_settings , selected_model , name1 , name2 , context , check , history_size ) :
yield i
2023-01-15 10:20:04 -05:00
def remove_last_message ( name1 , name2 ) :
2023-01-26 22:40:39 -05:00
if not history [ ' internal ' ] [ - 1 ] [ 0 ] == ' <|BEGIN-VISIBLE-CHAT|> ' :
last = history [ ' visible ' ] . pop ( )
history [ ' internal ' ] . pop ( )
else :
last = [ ' ' , ' ' ]
2023-01-15 10:20:04 -05:00
if args . cai_chat :
2023-01-26 22:40:39 -05:00
return generate_chat_html ( history [ ' visible ' ] , name1 , name2 , character ) , last [ 0 ]
2023-01-15 10:20:04 -05:00
else :
2023-01-26 22:40:39 -05:00
return history [ ' visible ' ] , last [ 0 ]
2023-01-15 01:19:09 -05:00
2023-01-15 10:20:04 -05:00
def clear_html ( ) :
2023-01-19 14:46:46 -05:00
return generate_chat_html ( [ ] , " " , " " , character )
2023-01-15 10:20:04 -05:00
2023-01-25 08:52:35 -05:00
def clear_chat_log ( _character , name1 , name2 ) :
global history
if _character != ' None ' :
2023-01-26 22:40:39 -05:00
for i in range ( len ( history [ ' internal ' ] ) ) :
if ' <|BEGIN-VISIBLE-CHAT|> ' in history [ ' internal ' ] [ i ] [ 0 ] :
history [ ' visible ' ] = [ [ ' ' , history [ ' internal ' ] [ i ] [ 1 ] ] ]
history [ ' internal ' ] = history [ ' internal ' ] [ : i + 1 ]
break
2023-01-25 08:52:35 -05:00
else :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = [ ]
history [ ' visible ' ] = [ ]
2023-01-25 08:52:35 -05:00
if args . cai_chat :
2023-01-26 22:40:39 -05:00
return generate_chat_html ( history [ ' visible ' ] , name1 , name2 , character )
2023-01-25 08:52:35 -05:00
else :
2023-01-26 22:40:39 -05:00
return history [ ' visible ' ]
2023-01-25 08:52:35 -05:00
2023-01-19 12:03:47 -05:00
def redraw_html ( name1 , name2 ) :
global history
2023-01-26 22:40:39 -05:00
return generate_chat_html ( history [ ' visible ' ] , name1 , name2 , character )
2023-01-19 12:03:47 -05:00
2023-01-23 07:45:10 -05:00
def tokenize_dialogue ( dialogue , name1 , name2 ) :
2023-01-26 22:40:39 -05:00
_history = [ ]
2023-01-26 11:45:19 -05:00
2023-01-21 00:48:06 -05:00
dialogue = re . sub ( ' <START> ' , ' ' , dialogue )
2023-01-28 18:18:23 -05:00
dialogue = re . sub ( ' <start> ' , ' ' , dialogue )
2023-01-21 00:48:06 -05:00
dialogue = re . sub ( ' ( \n |^)[Aa]non: ' , ' \\ 1You: ' , dialogue )
2023-01-28 18:18:23 -05:00
dialogue = re . sub ( ' ( \n |^) \ [CHARACTER \ ]: ' , f ' \\ 1 { name2 } : ' , dialogue )
2023-01-21 00:48:06 -05:00
idx = [ m . start ( ) for m in re . finditer ( f " (^| \n )( { name1 } | { name2 } ): " , dialogue ) ]
2023-01-26 11:45:19 -05:00
if len ( idx ) == 0 :
2023-01-26 22:40:39 -05:00
return _history
2023-01-26 11:45:19 -05:00
2023-01-21 00:48:06 -05:00
messages = [ ]
for i in range ( len ( idx ) - 1 ) :
messages . append ( dialogue [ idx [ i ] : idx [ i + 1 ] ] . strip ( ) )
2023-01-23 12:28:02 -05:00
messages . append ( dialogue [ idx [ - 1 ] : ] . strip ( ) )
2023-01-21 00:48:06 -05:00
entry = [ ' ' , ' ' ]
for i in messages :
if i . startswith ( f ' { name1 } : ' ) :
entry [ 0 ] = i [ len ( f ' { name1 } : ' ) : ] . strip ( )
elif i . startswith ( f ' { name2 } : ' ) :
entry [ 1 ] = i [ len ( f ' { name2 } : ' ) : ] . strip ( )
if not ( len ( entry [ 0 ] ) == 0 and len ( entry [ 1 ] ) == 0 ) :
2023-01-26 22:40:39 -05:00
_history . append ( entry )
2023-01-21 00:48:06 -05:00
entry = [ ' ' , ' ' ]
2023-01-23 12:28:02 -05:00
2023-01-28 18:41:02 -05:00
print ( f " \n Dialogue tokenized to (formatted as [question, reply]): \n \n " , end = ' ' )
2023-01-28 18:18:23 -05:00
for i in _history :
print ( i )
print ( " -------------------- \n " , end = ' ' )
2023-01-26 22:40:39 -05:00
return _history
2023-01-21 00:48:06 -05:00
2023-01-23 07:45:10 -05:00
def save_history ( ) :
if not Path ( ' logs ' ) . exists ( ) :
Path ( ' logs ' ) . mkdir ( )
with open ( Path ( ' logs/conversation.json ' ) , ' w ' ) as f :
2023-01-27 00:16:05 -05:00
f . write ( json . dumps ( { ' data ' : history [ ' internal ' ] , ' data_visible ' : history [ ' visible ' ] } ) )
2023-01-23 07:45:10 -05:00
return Path ( ' logs/conversation.json ' )
2023-01-25 13:45:25 -05:00
def upload_history ( file , name1 , name2 ) :
2023-01-23 07:45:10 -05:00
global history
file = file . decode ( ' utf-8 ' )
try :
2023-01-23 13:29:01 -05:00
j = json . loads ( file )
if ' data ' in j :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = j [ ' data ' ]
2023-01-27 00:16:05 -05:00
if ' data_visible ' in j :
history [ ' visible ' ] = j [ ' data_visible ' ]
else :
2023-01-27 10:01:11 -05:00
history [ ' visible ' ] = copy . deepcopy ( history [ ' internal ' ] )
2023-01-23 13:29:01 -05:00
# Compatibility with Pygmalion AI's official web UI
elif ' chat ' in j :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = [ ' : ' . join ( x . split ( ' : ' ) [ 1 : ] ) . strip ( ) for x in j [ ' chat ' ] ]
2023-01-23 13:29:01 -05:00
if len ( j [ ' chat ' ] ) > 0 and j [ ' chat ' ] [ 0 ] . startswith ( f ' { name2 } : ' ) :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = [ [ ' <|BEGIN-VISIBLE-CHAT|> ' , history [ ' internal ' ] [ 0 ] ] ] + [ [ history [ ' internal ' ] [ i ] , history [ ' internal ' ] [ i + 1 ] ] for i in range ( 1 , len ( history [ ' internal ' ] ) - 1 , 2 ) ]
2023-01-23 13:29:01 -05:00
else :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = [ [ history [ ' internal ' ] [ i ] , history [ ' internal ' ] [ i + 1 ] ] for i in range ( 0 , len ( history [ ' internal ' ] ) - 1 , 2 ) ]
2023-01-23 07:45:10 -05:00
except :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = tokenize_dialogue ( file , name1 , name2 )
2023-01-27 10:01:11 -05:00
history [ ' visible ' ] = copy . deepcopy ( history [ ' internal ' ] )
2023-01-23 07:45:10 -05:00
2023-01-19 14:46:46 -05:00
def load_character ( _character , name1 , name2 ) :
global history , character
context = " "
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = [ ]
history [ ' visible ' ] = [ ]
2023-01-19 14:46:46 -05:00
if _character != ' None ' :
character = _character
with open ( Path ( f ' characters/ { _character } .json ' ) , ' r ' ) as f :
data = json . loads ( f . read ( ) )
name2 = data [ ' char_name ' ]
if ' char_persona ' in data and data [ ' char_persona ' ] != ' ' :
context + = f " { data [ ' char_name ' ] } ' s Persona: { data [ ' char_persona ' ] } \n "
if ' world_scenario ' in data and data [ ' world_scenario ' ] != ' ' :
context + = f " Scenario: { data [ ' world_scenario ' ] } \n "
2023-01-19 17:04:54 -05:00
context = f " { context . strip ( ) } \n <START> \n "
2023-01-19 14:46:46 -05:00
if ' example_dialogue ' in data and data [ ' example_dialogue ' ] != ' ' :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] = tokenize_dialogue ( data [ ' example_dialogue ' ] , name1 , name2 )
2023-01-21 00:48:06 -05:00
if ' char_greeting ' in data and len ( data [ ' char_greeting ' ] . strip ( ) ) > 0 :
2023-01-28 22:04:11 -05:00
history [ ' internal ' ] + = [ [ ' <|BEGIN-VISIBLE-CHAT|> ' , apply_extensions ( data [ ' char_greeting ' ] , " output " ) ] ]
history [ ' visible ' ] + = [ [ ' ' , apply_extensions ( data [ ' char_greeting ' ] , " output " ) ] ]
2023-01-21 00:48:06 -05:00
else :
2023-01-26 22:40:39 -05:00
history [ ' internal ' ] + = [ [ ' <|BEGIN-VISIBLE-CHAT|> ' , " Hello there! " ] ]
history [ ' visible ' ] + = [ [ ' ' , " Hello there! " ] ]
2023-01-19 14:46:46 -05:00
else :
character = None
context = settings [ ' context_pygmalion ' ]
name2 = settings [ ' name2_pygmalion ' ]
if args . cai_chat :
2023-01-26 22:40:39 -05:00
return name2 , context , generate_chat_html ( history [ ' visible ' ] , name1 , name2 , character )
2023-01-19 14:46:46 -05:00
else :
2023-01-26 22:40:39 -05:00
return name2 , context , history [ ' visible ' ]
2023-01-19 14:46:46 -05:00
2023-01-28 19:01:56 -05:00
def upload_character ( json_file , img , tavern = False ) :
2023-01-28 18:18:23 -05:00
json_file = json_file if type ( json_file ) == str else json_file . decode ( ' utf-8 ' )
2023-01-28 17:16:37 -05:00
data = json . loads ( json_file )
2023-01-25 13:45:25 -05:00
outfile_name = data [ " char_name " ]
i = 1
while Path ( f ' characters/ { outfile_name } .json ' ) . exists ( ) :
outfile_name = f ' { data [ " char_name " ] } _ { i : 03d } '
i + = 1
2023-01-28 19:01:56 -05:00
if tavern :
outfile_name = f ' TavernAI- { outfile_name } '
2023-01-25 13:45:25 -05:00
with open ( Path ( f ' characters/ { outfile_name } .json ' ) , ' w ' ) as f :
2023-01-28 17:16:37 -05:00
f . write ( json_file )
if img is not None :
2023-01-28 18:18:23 -05:00
img = Image . open ( io . BytesIO ( img ) )
img . save ( Path ( f ' characters/ { outfile_name } .png ' ) )
2023-01-25 13:45:25 -05:00
print ( f ' New character saved to " characters/ { outfile_name } .json " . ' )
return outfile_name
2023-01-28 18:18:23 -05:00
def upload_tavern_character ( img , name1 , name2 ) :
_img = Image . open ( io . BytesIO ( img ) )
_img . getexif ( )
decoded_string = base64 . b64decode ( _img . info [ ' chara ' ] )
_json = json . loads ( decoded_string )
_json = { " char_name " : _json [ ' name ' ] , " char_persona " : _json [ ' description ' ] , " char_greeting " : _json [ " first_mes " ] , " example_dialogue " : _json [ ' mes_example ' ] , " world_scenario " : _json [ ' scenario ' ] }
_json [ ' example_dialogue ' ] = _json [ ' example_dialogue ' ] . replace ( ' {{ user}} ' , name1 ) . replace ( ' {{ char}} ' , _json [ ' char_name ' ] )
2023-01-28 19:01:56 -05:00
return upload_character ( json . dumps ( _json ) , img , tavern = True )
2023-01-28 18:18:23 -05:00
2023-01-28 17:16:37 -05:00
def upload_your_profile_picture ( img ) :
2023-01-28 18:18:23 -05:00
img = Image . open ( io . BytesIO ( img ) )
img . save ( Path ( f ' img_me.png ' ) )
print ( f ' Profile picture saved to " img_me.png " ' )
2023-01-28 17:16:37 -05:00
2023-01-19 14:46:46 -05:00
suffix = ' _pygmalion ' if ' pygmalion ' in model_name . lower ( ) else ' '
2023-01-15 16:16:46 -05:00
with gr . Blocks ( css = css + " .h- \ [40vh \ ] { height: 66.67vh} .gradio-container { max-width: 800px; margin-left: auto; margin-right: auto} " , analytics_enabled = False ) as interface :
if args . cai_chat :
2023-01-19 14:46:46 -05:00
display1 = gr . HTML ( value = generate_chat_html ( [ ] , " " , " " , character ) )
2023-01-15 16:16:46 -05:00
else :
display1 = gr . Chatbot ( )
2023-01-23 12:04:01 -05:00
textbox = gr . Textbox ( label = ' Input ' )
2023-01-15 16:16:46 -05:00
btn = gr . Button ( " Generate " )
2023-01-09 15:23:43 -05:00
with gr . Row ( ) :
2023-01-18 20:44:47 -05:00
stop = gr . Button ( " Stop " )
2023-01-22 00:19:58 -05:00
btn_regenerate = gr . Button ( " Regenerate " )
btn_remove_last = gr . Button ( " Remove last " )
btn_clear = gr . Button ( " Clear history " )
2023-01-13 13:02:17 -05:00
2023-01-15 16:16:46 -05:00
with gr . Row ( ) :
2023-01-07 20:52:46 -05:00
with gr . Column ( ) :
2023-01-20 15:03:09 -05:00
length_slider = gr . Slider ( minimum = settings [ ' max_new_tokens_min ' ] , maximum = settings [ ' max_new_tokens_max ' ] , step = 1 , label = ' max_new_tokens ' , value = settings [ ' max_new_tokens ' ] )
2023-01-21 22:02:46 -05:00
with gr . Row ( ) :
model_menu = gr . Dropdown ( choices = available_models , value = model_name , label = ' Model ' )
create_refresh_button ( model_menu , lambda : None , lambda : { " choices " : get_available_models ( ) } , " refresh-button " )
2023-01-15 16:16:46 -05:00
with gr . Column ( ) :
2023-01-22 01:15:35 -05:00
history_size_slider = gr . Slider ( minimum = settings [ ' history_size_min ' ] , maximum = settings [ ' history_size_max ' ] , step = 1 , label = ' Chat history size in prompt (0 for no limit) ' , value = settings [ ' history_size ' ] )
2023-01-21 22:02:46 -05:00
with gr . Row ( ) :
2023-01-23 18:49:44 -05:00
preset_menu = gr . Dropdown ( choices = available_presets , value = settings [ f ' preset { suffix } ' ] , label = ' Generation parameters preset ' )
2023-01-21 22:02:46 -05:00
create_refresh_button ( preset_menu , lambda : None , lambda : { " choices " : get_available_presets ( ) } , " refresh-button " )
2023-01-15 16:16:46 -05:00
2023-01-19 16:58:45 -05:00
name1 = gr . Textbox ( value = settings [ f ' name1 { suffix } ' ] , lines = 1 , label = ' Your name ' )
name2 = gr . Textbox ( value = settings [ f ' name2 { suffix } ' ] , lines = 1 , label = ' Bot \' s name ' )
context = gr . Textbox ( value = settings [ f ' context { suffix } ' ] , lines = 2 , label = ' Context ' )
2023-01-15 16:16:46 -05:00
with gr . Row ( ) :
2023-01-21 22:02:46 -05:00
character_menu = gr . Dropdown ( choices = available_characters , value = " None " , label = ' Character ' )
create_refresh_button ( character_menu , lambda : None , lambda : { " choices " : get_available_characters ( ) } , " refresh-button " )
2023-01-19 14:46:46 -05:00
with gr . Row ( ) :
2023-01-19 16:58:45 -05:00
check = gr . Checkbox ( value = settings [ f ' stop_at_newline { suffix } ' ] , label = ' Stop generating at new line character? ' )
2023-01-19 12:03:47 -05:00
with gr . Row ( ) :
2023-01-28 18:28:08 -05:00
with gr . Tab ( ' Chat history ' ) :
with gr . Row ( ) :
with gr . Column ( ) :
gr . Markdown ( ' Upload ' )
upload = gr . File ( type = ' binary ' )
with gr . Column ( ) :
gr . Markdown ( ' Download ' )
download = gr . File ( )
save_btn = gr . Button ( value = " Click me " )
2023-01-25 13:45:25 -05:00
with gr . Tab ( ' Upload character ' ) :
2023-01-28 17:16:37 -05:00
with gr . Row ( ) :
with gr . Column ( ) :
gr . Markdown ( ' 1. Select the JSON file ' )
upload_char = gr . File ( type = ' binary ' )
with gr . Column ( ) :
gr . Markdown ( ' 2. Select your character \' s profile picture (optional) ' )
upload_img = gr . File ( type = ' binary ' )
upload_btn = gr . Button ( value = " Submit " )
with gr . Tab ( ' Upload your profile picture ' ) :
upload_img_me = gr . File ( type = ' binary ' )
2023-01-28 18:18:23 -05:00
with gr . Tab ( ' Upload TavernAI Character Card ' ) :
upload_img_tavern = gr . File ( type = ' binary ' )
2023-01-15 16:16:46 -05:00
2023-01-28 21:00:51 -05:00
if args . extensions is not None :
extensions_ui_elements = [ ]
2023-01-28 21:21:40 -05:00
default_values = [ ]
2023-01-28 21:00:51 -05:00
gr . Markdown ( ' ## Extensions parameters ' )
for ext in sorted ( extension_state , key = lambda x : extension_state [ x ] [ 1 ] ) :
if extension_state [ ext ] [ 0 ] == True :
params = eval ( f " extensions. { ext } .script.params " )
for param in params :
2023-01-28 21:21:40 -05:00
_id = f " { ext } - { param } "
default_value = settings [ _id ] if _id in settings else params [ param ]
default_values . append ( default_value )
2023-01-28 21:08:28 -05:00
if type ( params [ param ] ) == str :
2023-01-28 21:21:40 -05:00
extensions_ui_elements . append ( gr . Textbox ( value = default_value , label = f " { ext } - { param } " ) )
2023-01-28 21:08:28 -05:00
elif type ( params [ param ] ) in [ int , float ] :
2023-01-28 21:21:40 -05:00
extensions_ui_elements . append ( gr . Number ( value = default_value , label = f " { ext } - { param } " ) )
2023-01-28 21:08:28 -05:00
elif type ( params [ param ] ) == bool :
2023-01-28 21:21:40 -05:00
extensions_ui_elements . append ( gr . Checkbox ( value = default_value , label = f " { ext } - { param } " ) )
2023-01-28 21:00:51 -05:00
def update_extensions_parameters ( * kwargs ) :
i = 0
for ext in sorted ( extension_state , key = lambda x : extension_state [ x ] [ 1 ] ) :
if extension_state [ ext ] [ 0 ] == True :
params = eval ( f " extensions. { ext } .script.params " )
for param in params :
if len ( kwargs ) > = i + 1 :
params [ param ] = eval ( f " kwargs[ { i } ] " )
i + = 1
2023-01-28 21:21:40 -05:00
update_extensions_parameters ( * default_values )
2023-01-28 21:00:51 -05:00
btn_extensions = gr . Button ( " Apply " )
btn_extensions . click ( update_extensions_parameters , [ * extensions_ui_elements ] , [ ] )
2023-01-20 15:03:09 -05:00
input_params = [ textbox , length_slider , preset_menu , model_menu , name1 , name2 , context , check , history_size_slider ]
2023-01-15 10:20:04 -05:00
if args . cai_chat :
2023-01-20 15:03:09 -05:00
gen_event = btn . click ( cai_chatbot_wrapper , input_params , display1 , show_progress = args . no_stream , api_name = " textgen " )
gen_event2 = textbox . submit ( cai_chatbot_wrapper , input_params , display1 , show_progress = args . no_stream )
2023-01-15 10:20:04 -05:00
else :
2023-01-20 15:03:09 -05:00
gen_event = btn . click ( chatbot_wrapper , input_params , display1 , show_progress = args . no_stream , api_name = " textgen " )
gen_event2 = textbox . submit ( chatbot_wrapper , input_params , display1 , show_progress = args . no_stream )
2023-01-22 00:19:58 -05:00
gen_event3 = btn_regenerate . click ( regenerate_wrapper , input_params , display1 , show_progress = args . no_stream )
2023-01-15 10:20:04 -05:00
2023-01-25 08:52:35 -05:00
btn_clear . click ( clear_chat_log , [ character_menu , name1 , name2 ] , display1 )
2023-01-22 17:40:22 -05:00
btn_remove_last . click ( remove_last_message , [ name1 , name2 ] , [ display1 , textbox ] , show_progress = False )
2023-01-07 23:10:02 -05:00
btn . click ( lambda x : " " , textbox , textbox , show_progress = False )
2023-01-22 00:19:58 -05:00
btn_regenerate . click ( lambda x : " " , textbox , textbox , show_progress = False )
2023-01-07 23:33:45 -05:00
textbox . submit ( lambda x : " " , textbox , textbox , show_progress = False )
2023-01-22 00:19:58 -05:00
stop . click ( None , None , None , cancels = [ gen_event , gen_event2 , gen_event3 ] )
2023-01-19 12:03:47 -05:00
save_btn . click ( save_history , inputs = [ ] , outputs = [ download ] )
2023-01-19 14:46:46 -05:00
character_menu . change ( load_character , [ character_menu , name1 , name2 ] , [ name2 , context , display1 ] )
2023-01-25 13:45:25 -05:00
upload . upload ( upload_history , [ upload , name1 , name2 ] , [ ] )
2023-01-28 18:18:23 -05:00
upload_btn . click ( upload_character , [ upload_char , upload_img ] , [ character_menu ] )
upload_img_tavern . upload ( upload_tavern_character , [ upload_img_tavern , name1 , name2 ] , [ character_menu ] )
2023-01-28 17:16:37 -05:00
upload_img_me . upload ( upload_your_profile_picture , [ upload_img_me ] , [ ] )
2023-01-19 14:46:46 -05:00
2023-01-19 13:05:42 -05:00
if args . cai_chat :
upload . upload ( redraw_html , [ name1 , name2 ] , [ display1 ] )
2023-01-28 17:16:37 -05:00
upload_img_me . upload ( redraw_html , [ name1 , name2 ] , [ display1 ] )
2023-01-19 13:05:42 -05:00
else :
2023-01-26 22:40:39 -05:00
upload . upload ( lambda : history [ ' visible ' ] , [ ] , [ display1 ] )
2023-01-28 17:16:37 -05:00
upload_img_me . upload ( lambda : history [ ' visible ' ] , [ ] , [ display1 ] )
2023-01-19 12:03:47 -05:00
2023-01-18 20:44:47 -05:00
elif args . notebook :
with gr . Blocks ( css = css , analytics_enabled = False ) as interface :
gr . Markdown ( description )
with gr . Tab ( ' Raw ' ) :
textbox = gr . Textbox ( value = default_text , lines = 23 )
with gr . Tab ( ' Markdown ' ) :
markdown = gr . Markdown ( )
with gr . Tab ( ' HTML ' ) :
html = gr . HTML ( )
btn = gr . Button ( " Generate " )
stop = gr . Button ( " Stop " )
2023-01-10 23:33:57 -05:00
2023-01-18 20:44:47 -05:00
length_slider = gr . Slider ( minimum = settings [ ' max_new_tokens_min ' ] , maximum = settings [ ' max_new_tokens_max ' ] , step = 1 , label = ' max_new_tokens ' , value = settings [ ' max_new_tokens ' ] )
with gr . Row ( ) :
with gr . Column ( ) :
2023-01-21 22:02:46 -05:00
with gr . Row ( ) :
model_menu = gr . Dropdown ( choices = available_models , value = model_name , label = ' Model ' )
create_refresh_button ( model_menu , lambda : None , lambda : { " choices " : get_available_models ( ) } , " refresh-button " )
2023-01-18 20:44:47 -05:00
with gr . Column ( ) :
2023-01-21 22:02:46 -05:00
with gr . Row ( ) :
2023-01-23 18:49:44 -05:00
preset_menu = gr . Dropdown ( choices = available_presets , value = settings [ ' preset ' ] , label = ' Generation parameters preset ' )
2023-01-21 22:02:46 -05:00
create_refresh_button ( preset_menu , lambda : None , lambda : { " choices " : get_available_presets ( ) } , " refresh-button " )
2023-01-18 20:44:47 -05:00
2023-01-18 21:56:42 -05:00
gen_event = btn . click ( generate_reply , [ textbox , length_slider , preset_menu , model_menu ] , [ textbox , markdown , html ] , show_progress = args . no_stream , api_name = " textgen " )
gen_event2 = textbox . submit ( generate_reply , [ textbox , length_slider , preset_menu , model_menu ] , [ textbox , markdown , html ] , show_progress = args . no_stream )
2023-01-18 20:44:47 -05:00
stop . click ( None , None , None , cancels = [ gen_event , gen_event2 ] )
else :
2023-01-08 18:10:31 -05:00
with gr . Blocks ( css = css , analytics_enabled = False ) as interface :
gr . Markdown ( description )
2023-01-06 20:05:37 -05:00
with gr . Row ( ) :
with gr . Column ( ) :
textbox = gr . Textbox ( value = default_text , lines = 15 , label = ' Input ' )
2023-01-15 13:23:41 -05:00
length_slider = gr . Slider ( minimum = settings [ ' max_new_tokens_min ' ] , maximum = settings [ ' max_new_tokens_max ' ] , step = 1 , label = ' max_new_tokens ' , value = settings [ ' max_new_tokens ' ] )
2023-01-21 22:02:46 -05:00
with gr . Row ( ) :
2023-01-23 18:49:44 -05:00
preset_menu = gr . Dropdown ( choices = available_presets , value = settings [ ' preset ' ] , label = ' Generation parameters preset ' )
2023-01-21 22:02:46 -05:00
create_refresh_button ( preset_menu , lambda : None , lambda : { " choices " : get_available_presets ( ) } , " refresh-button " )
with gr . Row ( ) :
model_menu = gr . Dropdown ( choices = available_models , value = model_name , label = ' Model ' )
create_refresh_button ( model_menu , lambda : None , lambda : { " choices " : get_available_models ( ) } , " refresh-button " )
2023-01-06 20:05:37 -05:00
btn = gr . Button ( " Generate " )
2023-01-18 20:44:47 -05:00
with gr . Row ( ) :
with gr . Column ( ) :
cont = gr . Button ( " Continue " )
with gr . Column ( ) :
stop = gr . Button ( " Stop " )
2023-01-06 20:05:37 -05:00
with gr . Column ( ) :
with gr . Tab ( ' Raw ' ) :
2023-01-10 23:36:11 -05:00
output_textbox = gr . Textbox ( lines = 15 , label = ' Output ' )
2023-01-06 20:05:37 -05:00
with gr . Tab ( ' Markdown ' ) :
markdown = gr . Markdown ( )
2023-01-06 21:14:08 -05:00
with gr . Tab ( ' HTML ' ) :
html = gr . HTML ( )
2023-01-06 20:05:37 -05:00
2023-01-18 21:56:42 -05:00
gen_event = btn . click ( generate_reply , [ textbox , length_slider , preset_menu , model_menu ] , [ output_textbox , markdown , html ] , show_progress = args . no_stream , api_name = " textgen " )
gen_event2 = textbox . submit ( generate_reply , [ textbox , length_slider , preset_menu , model_menu ] , [ output_textbox , markdown , html ] , show_progress = args . no_stream )
cont_event = cont . click ( generate_reply , [ output_textbox , length_slider , preset_menu , model_menu ] , [ output_textbox , markdown , html ] , show_progress = args . no_stream )
2023-01-18 20:44:47 -05:00
stop . click ( None , None , None , cancels = [ gen_event , gen_event2 , cont_event ] )
2022-12-21 11:27:31 -05:00
2023-01-25 14:10:35 -05:00
interface . queue ( )
2023-01-20 21:45:16 -05:00
if args . listen :
2023-01-27 13:38:34 -05:00
interface . launch ( share = args . share , server_name = " 0.0.0.0 " , server_port = args . listen_port )
2023-01-20 21:45:16 -05:00
else :
2023-01-28 18:35:05 -05:00
interface . launch ( share = args . share , server_port = args . listen_port )