chat.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. import base64
  2. import copy
  3. import io
  4. import json
  5. import re
  6. from datetime import datetime
  7. from pathlib import Path
  8. from PIL import Image
  9. import modules.shared as shared
  10. import modules.extensions as extensions_module
  11. from modules.extensions import apply_extensions
  12. from modules.html_generator import generate_chat_html
  13. from modules.text_generation import encode, generate_reply, get_max_prompt_length
  14. # This gets the new line characters right.
  15. def clean_chat_message(text):
  16. text = text.replace('\n', '\n\n')
  17. text = re.sub(r"\n{3,}", "\n\n", text)
  18. text = text.strip()
  19. return text
  20. def generate_chat_prompt(user_input, max_new_tokens, name1, name2, context, chat_prompt_size, impersonate=False):
  21. user_input = clean_chat_message(user_input)
  22. rows = [f"{context.strip()}\n"]
  23. if shared.soft_prompt:
  24. chat_prompt_size -= shared.soft_prompt_tensor.shape[1]
  25. max_length = min(get_max_prompt_length(max_new_tokens), chat_prompt_size)
  26. i = len(shared.history['internal'])-1
  27. while i >= 0 and len(encode(''.join(rows), max_new_tokens)[0]) < max_length:
  28. rows.insert(1, f"{name2}: {shared.history['internal'][i][1].strip()}\n")
  29. if not (shared.history['internal'][i][0] == '<|BEGIN-VISIBLE-CHAT|>'):
  30. rows.insert(1, f"{name1}: {shared.history['internal'][i][0].strip()}\n")
  31. i -= 1
  32. if not impersonate:
  33. rows.append(f"{name1}: {user_input}\n")
  34. rows.append(apply_extensions(f"{name2}:", "bot_prefix"))
  35. limit = 3
  36. else:
  37. rows.append(f"{name1}:")
  38. limit = 2
  39. while len(rows) > limit and len(encode(''.join(rows), max_new_tokens)[0]) >= max_length:
  40. rows.pop(1)
  41. prompt = ''.join(rows)
  42. return prompt
  43. def extract_message_from_reply(question, reply, current, other, check, extensions=False):
  44. next_character_found = False
  45. substring_found = False
  46. previous_idx = [m.start() for m in re.finditer(f"(^|\n){re.escape(current)}:", question)]
  47. idx = [m.start() for m in re.finditer(f"(^|\n){re.escape(current)}:", reply)]
  48. idx = idx[len(previous_idx)-1]
  49. if extensions:
  50. reply = reply[idx + 1 + len(apply_extensions(f"{current}:", "bot_prefix")):]
  51. else:
  52. reply = reply[idx + 1 + len(f"{current}:"):]
  53. if check:
  54. reply = reply.split('\n')[0].strip()
  55. else:
  56. idx = reply.find(f"\n{other}:")
  57. if idx != -1:
  58. reply = reply[:idx]
  59. next_character_found = True
  60. reply = clean_chat_message(reply)
  61. # Detect if something like "\nYo" is generated just before
  62. # "\nYou:" is completed
  63. tmp = f"\n{other}:"
  64. for j in range(1, len(tmp)):
  65. if reply[-j:] == tmp[:j]:
  66. substring_found = True
  67. return reply, next_character_found, substring_found
  68. def stop_everything_event():
  69. shared.stop_everything = True
  70. def chatbot_wrapper(text, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, name1, name2, context, check, chat_prompt_size):
  71. shared.stop_everything = False
  72. # Check if any extension wants to hijack this function call
  73. visible_text = None
  74. prompt = None
  75. for extension, _ in extensions_module.iterator():
  76. if hasattr(extension, 'input_hijack') and extension.input_hijack['state'] == True:
  77. extension.input_hijack['state'] = False
  78. values = extension.input_hijack['value']
  79. if len(values) == 2:
  80. text, visible_text = values
  81. elif len(values) == 4:
  82. text, visible_text, reply, visible_reply = valueso
  83. if not shared.stop_everything:
  84. shared.history['internal'].append([text, reply])
  85. shared.history['visible'].append([visible_text, visible_reply])
  86. return shared.history['visible']
  87. if hasattr(extension, 'prompt_hijack') and extension.prompt_hijack['state'] == True:
  88. prompt = extension.prompt_hijack['value']
  89. just_started = True
  90. eos_token = '\n' if check else None
  91. if 'pygmalion' in shared.model_name.lower():
  92. name1 = "You"
  93. if visible_text is None:
  94. visible_text = text
  95. if shared.args.chat:
  96. visible_text = visible_text.replace('\n', '<br>')
  97. text = apply_extensions(text, "input")
  98. if prompt is None:
  99. prompt = generate_chat_prompt(text, max_new_tokens, name1, name2, context, chat_prompt_size)
  100. # Generate
  101. for reply in generate_reply(prompt, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, eos_token=eos_token, stopping_string=f"\n{name1}:"):
  102. # Extracting the reply
  103. reply, next_character_found, substring_found = extract_message_from_reply(prompt, reply, name2, name1, check, extensions=True)
  104. visible_reply = apply_extensions(reply, "output")
  105. if shared.args.chat:
  106. visible_reply = visible_reply.replace('\n', '<br>')
  107. # We need this global variable to handle the Stop event,
  108. # otherwise gradio gets confused
  109. if shared.stop_everything:
  110. return shared.history['visible']
  111. if just_started:
  112. just_started = False
  113. shared.history['internal'].append(['', ''])
  114. shared.history['visible'].append(['', ''])
  115. shared.history['internal'][-1] = [text, reply]
  116. shared.history['visible'][-1] = [visible_text, visible_reply]
  117. if not substring_found:
  118. yield shared.history['visible']
  119. if next_character_found:
  120. break
  121. yield shared.history['visible']
  122. def impersonate_wrapper(text, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, name1, name2, context, check, chat_prompt_size):
  123. eos_token = '\n' if check else None
  124. if 'pygmalion' in shared.model_name.lower():
  125. name1 = "You"
  126. prompt = generate_chat_prompt(text, max_new_tokens, name1, name2, context, chat_prompt_size, impersonate=True)
  127. for reply in generate_reply(prompt, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, eos_token=eos_token, stopping_string=f"\n{name2}:"):
  128. reply, next_character_found, substring_found = extract_message_from_reply(prompt, reply, name1, name2, check, extensions=False)
  129. if not substring_found:
  130. yield reply
  131. if next_character_found:
  132. break
  133. yield reply
  134. def cai_chatbot_wrapper(text, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, name1, name2, context, check, chat_prompt_size):
  135. for _history in chatbot_wrapper(text, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, name1, name2, context, check, chat_prompt_size):
  136. yield generate_chat_html(_history, name1, name2, shared.character)
  137. def regenerate_wrapper(text, max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, name1, name2, context, check, chat_prompt_size):
  138. if shared.character != 'None' and len(shared.history['visible']) == 1:
  139. if shared.args.cai_chat:
  140. yield generate_chat_html(shared.history['visible'], name1, name2, shared.character)
  141. else:
  142. yield shared.history['visible']
  143. else:
  144. last_visible = shared.history['visible'].pop()
  145. last_internal = shared.history['internal'].pop()
  146. for _history in chatbot_wrapper(last_internal[0], max_new_tokens, do_sample, temperature, top_p, typical_p, repetition_penalty, top_k, min_length, no_repeat_ngram_size, num_beams, penalty_alpha, length_penalty, early_stopping, name1, name2, context, check, chat_prompt_size):
  147. if shared.args.cai_chat:
  148. shared.history['visible'][-1] = [last_visible[0], _history[-1][1]]
  149. yield generate_chat_html(shared.history['visible'], name1, name2, shared.character)
  150. else:
  151. shared.history['visible'][-1] = (last_visible[0], _history[-1][1])
  152. yield shared.history['visible']
  153. def remove_last_message(name1, name2):
  154. if not shared.history['internal'][-1][0] == '<|BEGIN-VISIBLE-CHAT|>':
  155. last = shared.history['visible'].pop()
  156. shared.history['internal'].pop()
  157. else:
  158. last = ['', '']
  159. if shared.args.cai_chat:
  160. return generate_chat_html(shared.history['visible'], name1, name2, shared.character), last[0]
  161. else:
  162. return shared.history['visible'], last[0]
  163. def send_last_reply_to_input():
  164. if len(shared.history['internal']) > 0:
  165. return shared.history['internal'][-1][1]
  166. else:
  167. return ''
  168. def replace_last_reply(text, name1, name2):
  169. if len(shared.history['visible']) > 0:
  170. if shared.args.cai_chat:
  171. shared.history['visible'][-1][1] = text
  172. else:
  173. shared.history['visible'][-1] = (shared.history['visible'][-1][0], text)
  174. shared.history['internal'][-1][1] = apply_extensions(text, "input")
  175. if shared.args.cai_chat:
  176. return generate_chat_html(shared.history['visible'], name1, name2, shared.character)
  177. else:
  178. return shared.history['visible']
  179. def clear_html():
  180. return generate_chat_html([], "", "", shared.character)
  181. def clear_chat_log(name1, name2):
  182. if shared.character != 'None':
  183. for i in range(len(shared.history['internal'])):
  184. if '<|BEGIN-VISIBLE-CHAT|>' in shared.history['internal'][i][0]:
  185. shared.history['visible'] = [['', apply_extensions(shared.history['internal'][i][1], "output")]]
  186. shared.history['internal'] = shared.history['internal'][:i+1]
  187. break
  188. else:
  189. shared.history['internal'] = []
  190. shared.history['visible'] = []
  191. if shared.args.cai_chat:
  192. return generate_chat_html(shared.history['visible'], name1, name2, shared.character)
  193. else:
  194. return shared.history['visible']
  195. def redraw_html(name1, name2):
  196. return generate_chat_html(shared.history['visible'], name1, name2, shared.character)
  197. def tokenize_dialogue(dialogue, name1, name2):
  198. _history = []
  199. dialogue = re.sub('<START>', '', dialogue)
  200. dialogue = re.sub('<start>', '', dialogue)
  201. dialogue = re.sub('(\n|^)[Aa]non:', '\\1You:', dialogue)
  202. dialogue = re.sub('(\n|^)\[CHARACTER\]:', f'\\g<1>{name2}:', dialogue)
  203. idx = [m.start() for m in re.finditer(f"(^|\n)({re.escape(name1)}|{re.escape(name2)}):", dialogue)]
  204. if len(idx) == 0:
  205. return _history
  206. messages = []
  207. for i in range(len(idx)-1):
  208. messages.append(dialogue[idx[i]:idx[i+1]].strip())
  209. messages.append(dialogue[idx[-1]:].strip())
  210. entry = ['', '']
  211. for i in messages:
  212. if i.startswith(f'{name1}:'):
  213. entry[0] = i[len(f'{name1}:'):].strip()
  214. elif i.startswith(f'{name2}:'):
  215. entry[1] = i[len(f'{name2}:'):].strip()
  216. if not (len(entry[0]) == 0 and len(entry[1]) == 0):
  217. _history.append(entry)
  218. entry = ['', '']
  219. print("\033[1;32;1m\nDialogue tokenized to:\033[0;37;0m\n", end='')
  220. for row in _history:
  221. for column in row:
  222. print("\n")
  223. for line in column.strip().split('\n'):
  224. print("| "+line+"\n")
  225. print("|\n")
  226. print("------------------------------")
  227. return _history
  228. def save_history(timestamp=True):
  229. prefix = '' if shared.character == 'None' else f"{shared.character}_"
  230. if timestamp:
  231. fname = f"{prefix}{datetime.now().strftime('%Y%m%d-%H%M%S')}.json"
  232. else:
  233. fname = f"{prefix}persistent.json"
  234. if not Path('logs').exists():
  235. Path('logs').mkdir()
  236. with open(Path(f'logs/{fname}'), 'w') as f:
  237. f.write(json.dumps({'data': shared.history['internal'], 'data_visible': shared.history['visible']}, indent=2))
  238. return Path(f'logs/{fname}')
  239. def load_history(file, name1, name2):
  240. file = file.decode('utf-8')
  241. try:
  242. j = json.loads(file)
  243. if 'data' in j:
  244. shared.history['internal'] = j['data']
  245. if 'data_visible' in j:
  246. shared.history['visible'] = j['data_visible']
  247. else:
  248. shared.history['visible'] = copy.deepcopy(shared.history['internal'])
  249. # Compatibility with Pygmalion AI's official web UI
  250. elif 'chat' in j:
  251. shared.history['internal'] = [':'.join(x.split(':')[1:]).strip() for x in j['chat']]
  252. if len(j['chat']) > 0 and j['chat'][0].startswith(f'{name2}:'):
  253. shared.history['internal'] = [['<|BEGIN-VISIBLE-CHAT|>', shared.history['internal'][0]]] + [[shared.history['internal'][i], shared.history['internal'][i+1]] for i in range(1, len(shared.history['internal'])-1, 2)]
  254. shared.history['visible'] = copy.deepcopy(shared.history['internal'])
  255. shared.history['visible'][0][0] = ''
  256. else:
  257. shared.history['internal'] = [[shared.history['internal'][i], shared.history['internal'][i+1]] for i in range(0, len(shared.history['internal'])-1, 2)]
  258. shared.history['visible'] = copy.deepcopy(shared.history['internal'])
  259. except:
  260. shared.history['internal'] = tokenize_dialogue(file, name1, name2)
  261. shared.history['visible'] = copy.deepcopy(shared.history['internal'])
  262. def load_default_history(name1, name2):
  263. if Path('logs/persistent.json').exists():
  264. load_history(open(Path('logs/persistent.json'), 'rb').read(), name1, name2)
  265. else:
  266. shared.history['internal'] = []
  267. shared.history['visible'] = []
  268. def load_character(_character, name1, name2):
  269. context = ""
  270. shared.history['internal'] = []
  271. shared.history['visible'] = []
  272. if _character != 'None':
  273. shared.character = _character
  274. data = json.loads(open(Path(f'characters/{_character}.json'), 'r').read())
  275. name2 = data['char_name']
  276. if 'char_persona' in data and data['char_persona'] != '':
  277. context += f"{data['char_name']}'s Persona: {data['char_persona']}\n"
  278. if 'world_scenario' in data and data['world_scenario'] != '':
  279. context += f"Scenario: {data['world_scenario']}\n"
  280. context = f"{context.strip()}\n<START>\n"
  281. if 'example_dialogue' in data and data['example_dialogue'] != '':
  282. shared.history['internal'] = tokenize_dialogue(data['example_dialogue'], name1, name2)
  283. if 'char_greeting' in data and len(data['char_greeting'].strip()) > 0:
  284. shared.history['internal'] += [['<|BEGIN-VISIBLE-CHAT|>', data['char_greeting']]]
  285. shared.history['visible'] += [['', apply_extensions(data['char_greeting'], "output")]]
  286. else:
  287. shared.history['internal'] += [['<|BEGIN-VISIBLE-CHAT|>', "Hello there!"]]
  288. shared.history['visible'] += [['', "Hello there!"]]
  289. else:
  290. shared.character = None
  291. context = shared.settings['context_pygmalion']
  292. name2 = shared.settings['name2_pygmalion']
  293. if Path(f'logs/{shared.character}_persistent.json').exists():
  294. load_history(open(Path(f'logs/{shared.character}_persistent.json'), 'rb').read(), name1, name2)
  295. if shared.args.cai_chat:
  296. return name2, context, generate_chat_html(shared.history['visible'], name1, name2, shared.character)
  297. else:
  298. return name2, context, shared.history['visible']
  299. def upload_character(json_file, img, tavern=False):
  300. json_file = json_file if type(json_file) == str else json_file.decode('utf-8')
  301. data = json.loads(json_file)
  302. outfile_name = data["char_name"]
  303. i = 1
  304. while Path(f'characters/{outfile_name}.json').exists():
  305. outfile_name = f'{data["char_name"]}_{i:03d}'
  306. i += 1
  307. if tavern:
  308. outfile_name = f'TavernAI-{outfile_name}'
  309. with open(Path(f'characters/{outfile_name}.json'), 'w') as f:
  310. f.write(json_file)
  311. if img is not None:
  312. img = Image.open(io.BytesIO(img))
  313. img.save(Path(f'characters/{outfile_name}.png'))
  314. print(f'New character saved to "characters/{outfile_name}.json".')
  315. return outfile_name
  316. def upload_tavern_character(img, name1, name2):
  317. _img = Image.open(io.BytesIO(img))
  318. _img.getexif()
  319. decoded_string = base64.b64decode(_img.info['chara'])
  320. _json = json.loads(decoded_string)
  321. _json = {"char_name": _json['name'], "char_persona": _json['description'], "char_greeting": _json["first_mes"], "example_dialogue": _json['mes_example'], "world_scenario": _json['scenario']}
  322. _json['example_dialogue'] = _json['example_dialogue'].replace('{{user}}', name1).replace('{{char}}', _json['char_name'])
  323. return upload_character(json.dumps(_json), img, tavern=True)
  324. def upload_your_profile_picture(img):
  325. img = Image.open(io.BytesIO(img))
  326. img.save(Path('img_me.png'))
  327. print('Profile picture saved to "img_me.png"')