text-generation-webui/modules/extensions.py

134 lines
4.2 KiB
Python
Raw Normal View History

2023-03-24 15:51:27 -04:00
import traceback
2023-04-23 19:32:22 -04:00
from functools import partial
2023-03-24 15:51:27 -04:00
2023-03-15 14:11:16 -04:00
import gradio as gr
import extensions
2023-02-23 12:41:42 -05:00
import modules.shared as shared
2023-02-24 08:01:21 -05:00
state = {}
available_extensions = []
setup_called = set()
def apply_settings(extension, name):
if not hasattr(extension, 'params'):
return
for param in extension.params:
_id = f"{name}-{param}"
if _id not in shared.settings:
continue
extension.params[param] = shared.settings[_id]
2023-02-23 12:49:02 -05:00
def load_extensions():
2023-04-07 11:20:57 -04:00
global state, setup_called
2023-02-24 08:01:21 -05:00
for i, name in enumerate(shared.args.extensions):
if name in available_extensions:
if name != 'api':
print(f'Loading the extension "{name}"... ', end='')
2023-03-15 22:29:56 -04:00
try:
exec(f"import extensions.{name}.script")
extension = getattr(extensions, name).script
apply_settings(extension, name)
2023-04-07 11:20:57 -04:00
if extension not in setup_called and hasattr(extension, "setup"):
setup_called.add(extension)
extension.setup()
2023-04-24 23:10:21 -04:00
2023-03-15 22:29:56 -04:00
state[name] = [True, i]
if name != 'api':
print('Ok.')
2023-03-15 22:29:56 -04:00
except:
if name != 'api':
print('Fail.')
2023-04-24 23:10:21 -04:00
2023-03-24 15:51:27 -04:00
traceback.print_exc()
2023-02-23 12:49:02 -05:00
2023-04-07 11:20:57 -04:00
# This iterator returns the extensions in the order specified in the command-line
2023-02-24 08:01:21 -05:00
def iterator():
for name in sorted(state, key=lambda x: state[x][1]):
2023-04-06 23:52:02 -04:00
if state[name][0]:
yield getattr(extensions, name).script, name
2023-02-24 08:01:21 -05:00
2023-04-07 11:20:57 -04:00
# Extension functions that map string -> string
2023-04-23 19:32:22 -04:00
def _apply_string_extensions(function_name, text):
2023-02-24 08:01:21 -05:00
for extension, _ in iterator():
2023-04-23 19:32:22 -04:00
if hasattr(extension, function_name):
text = getattr(extension, function_name)(text)
2023-04-24 23:10:21 -04:00
return text
2023-04-23 19:32:22 -04:00
# Input hijack of extensions
def _apply_input_hijack(text, visible_text):
for extension, _ in iterator():
if hasattr(extension, 'input_hijack') and extension.input_hijack['state']:
extension.input_hijack['state'] = False
if callable(extension.input_hijack['value']):
text, visible_text = extension.input_hijack['value'](text, visible_text)
else:
text, visible_text = extension.input_hijack['value']
2023-04-24 23:10:21 -04:00
2023-04-23 19:32:22 -04:00
return text, visible_text
# custom_generate_chat_prompt handling
def _apply_custom_generate_chat_prompt(text, state, **kwargs):
custom_generate_chat_prompt = None
for extension, _ in iterator():
if custom_generate_chat_prompt is None and hasattr(extension, 'custom_generate_chat_prompt'):
custom_generate_chat_prompt = extension.custom_generate_chat_prompt
2023-04-24 23:10:21 -04:00
2023-04-23 19:32:22 -04:00
if custom_generate_chat_prompt is not None:
return custom_generate_chat_prompt(text, state, **kwargs)
2023-04-24 23:10:21 -04:00
2023-04-23 19:32:22 -04:00
return None
# Extension functions that override the default tokenizer output
def _apply_tokenizer_extensions(function_name, state, prompt, input_ids, input_embeds):
for extension, _ in iterator():
if hasattr(extension, function_name):
prompt, input_ids, input_embeds = getattr(extension, function_name)(state, prompt, input_ids, input_embeds)
2023-04-24 23:10:21 -04:00
2023-04-23 19:32:22 -04:00
return prompt, input_ids, input_embeds
EXTENSION_MAP = {
"input": partial(_apply_string_extensions, "input_modifier"),
"output": partial(_apply_string_extensions, "output_modifier"),
"bot_prefix": partial(_apply_string_extensions, "bot_prefix_modifier"),
"tokenizer": partial(_apply_tokenizer_extensions, "tokenizer_modifier"),
"input_hijack": _apply_input_hijack,
"custom_generate_chat_prompt": _apply_custom_generate_chat_prompt
}
def apply_extensions(typ, *args, **kwargs):
if typ not in EXTENSION_MAP:
raise ValueError(f"Invalid extension type {typ}")
2023-04-24 23:10:21 -04:00
2023-04-23 19:32:22 -04:00
return EXTENSION_MAP[typ](*args, **kwargs)
2023-02-23 12:55:21 -05:00
def create_extensions_block():
2023-03-19 09:25:49 -04:00
global setup_called
2023-03-24 15:51:27 -04:00
should_display_ui = False
for extension, name in iterator():
if hasattr(extension, "ui"):
should_display_ui = True
break
2023-03-19 09:22:24 -04:00
2023-02-24 17:00:11 -05:00
# Creating the extension ui elements
if should_display_ui:
2023-03-26 21:20:30 -04:00
with gr.Column(elem_id="extensions"):
2023-03-15 22:29:56 -04:00
for extension, name in iterator():
2023-03-26 21:20:30 -04:00
gr.Markdown(f"\n### {name}")
2023-03-15 22:29:56 -04:00
if hasattr(extension, "ui"):
extension.ui()