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
|
|
|
|
|
2023-02-23 10:05:25 -05:00
|
|
|
import extensions
|
2023-02-23 12:41:42 -05:00
|
|
|
import modules.shared as shared
|
2023-05-21 21:42:34 -04:00
|
|
|
from modules.logging_colors import logger
|
2023-07-03 23:03:30 -04:00
|
|
|
from inspect import signature
|
|
|
|
|
2023-02-23 10:05:25 -05:00
|
|
|
|
2023-02-24 08:01:21 -05:00
|
|
|
state = {}
|
2023-02-23 10:05:25 -05:00
|
|
|
available_extensions = []
|
2023-03-28 22:27:02 -04:00
|
|
|
setup_called = set()
|
2023-02-23 10:05:25 -05:00
|
|
|
|
2023-04-06 23:15:45 -04:00
|
|
|
|
2023-04-24 23:23:11 -04:00
|
|
|
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:
|
2023-04-23 14:52:43 -04:00
|
|
|
if name != 'api':
|
2023-05-21 21:42:34 -04:00
|
|
|
logger.info(f'Loading the extension "{name}"...')
|
2023-03-15 22:29:56 -04:00
|
|
|
try:
|
|
|
|
exec(f"import extensions.{name}.script")
|
2023-04-16 00:36:50 -04:00
|
|
|
extension = getattr(extensions, name).script
|
2023-04-25 00:16:23 -04:00
|
|
|
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]
|
|
|
|
except:
|
2023-05-21 21:42:34 -04:00
|
|
|
logger.error(f'Failed to load the extension "{name}".')
|
2023-03-24 15:51:27 -04:00
|
|
|
traceback.print_exc()
|
2023-02-23 12:49:02 -05:00
|
|
|
|
2023-04-06 23:15:45 -04: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():
|
2023-04-06 23:15:45 -04:00
|
|
|
for name in sorted(state, key=lambda x: state[x][1]):
|
2023-04-06 23:52:02 -04:00
|
|
|
if state[name][0]:
|
2023-04-16 00:36:50 -04:00
|
|
|
yield getattr(extensions, name).script, name
|
2023-02-24 08:01:21 -05:00
|
|
|
|
2023-04-06 23:15:45 -04:00
|
|
|
|
2023-04-07 11:20:57 -04:00
|
|
|
# Extension functions that map string -> string
|
2023-07-03 23:03:30 -04:00
|
|
|
def _apply_string_extensions(function_name, text, state):
|
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):
|
2023-07-03 23:03:30 -04:00
|
|
|
func = getattr(extension, function_name)
|
|
|
|
if len(signature(func).parameters) == 2:
|
|
|
|
text = func(text, state)
|
|
|
|
else:
|
|
|
|
text = func(text)
|
2023-04-24 23:10:21 -04:00
|
|
|
|
2023-02-23 10:05:25 -05:00
|
|
|
return text
|
|
|
|
|
2023-04-06 23:15:45 -04:00
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2023-05-09 19:18:02 -04:00
|
|
|
# custom_generate_chat_prompt handling - currently only the first one will work
|
2023-04-23 19:32:22 -04:00
|
|
|
def _apply_custom_generate_chat_prompt(text, state, **kwargs):
|
|
|
|
for extension, _ in iterator():
|
2023-05-09 19:18:02 -04:00
|
|
|
if hasattr(extension, 'custom_generate_chat_prompt'):
|
2023-05-10 10:29:59 -04:00
|
|
|
return extension.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
|
|
|
|
|
|
|
|
|
2023-05-05 17:53:03 -04:00
|
|
|
# Extension that modifies the input parameters before they are used
|
|
|
|
def _apply_state_modifier_extensions(state):
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, "state_modifier"):
|
|
|
|
state = getattr(extension, "state_modifier")(state)
|
|
|
|
|
|
|
|
return state
|
2023-05-09 21:49:39 -04:00
|
|
|
|
2023-05-05 17:53:03 -04:00
|
|
|
|
2023-05-21 12:24:54 -04:00
|
|
|
# Extension that modifies the chat history before it is used
|
|
|
|
def _apply_history_modifier_extensions(history):
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, "history_modifier"):
|
|
|
|
history = getattr(extension, "history_modifier")(history)
|
|
|
|
|
|
|
|
return history
|
|
|
|
|
|
|
|
|
2023-07-13 16:22:41 -04:00
|
|
|
# Extension functions that override the default tokenizer output - The order of execution is not defined
|
2023-04-23 19:32:22 -04:00
|
|
|
def _apply_tokenizer_extensions(function_name, state, prompt, input_ids, input_embeds):
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, function_name):
|
2023-07-13 16:22:41 -04:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2023-07-13 16:22:41 -04:00
|
|
|
# Allow extensions to add their own logits processors to the stack being run.
|
|
|
|
# Each extension would call `processor_list.append({their LogitsProcessor}())`.
|
|
|
|
def _apply_logits_processor_extensions(function_name, processor_list, input_ids):
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, function_name):
|
|
|
|
getattr(extension, function_name)(processor_list, input_ids)
|
|
|
|
|
|
|
|
|
2023-05-09 19:18:02 -04:00
|
|
|
# Get prompt length in tokens after applying extension functions which override the default tokenizer output
|
|
|
|
# currently only the first one will work
|
|
|
|
def _apply_custom_tokenized_length(prompt):
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, 'custom_tokenized_length'):
|
|
|
|
return getattr(extension, 'custom_tokenized_length')(prompt)
|
2023-05-09 21:49:39 -04:00
|
|
|
|
2023-05-09 19:18:02 -04:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
# Custom generate reply handling - currently only the first one will work
|
2023-05-05 17:53:03 -04:00
|
|
|
def _apply_custom_generate_reply():
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, 'custom_generate_reply'):
|
|
|
|
return getattr(extension, 'custom_generate_reply')
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2023-05-16 23:03:39 -04:00
|
|
|
def _apply_custom_css():
|
|
|
|
all_css = ''
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, 'custom_css'):
|
|
|
|
all_css += getattr(extension, 'custom_css')()
|
|
|
|
|
|
|
|
return all_css
|
|
|
|
|
|
|
|
|
|
|
|
def _apply_custom_js():
|
|
|
|
all_js = ''
|
|
|
|
for extension, _ in iterator():
|
|
|
|
if hasattr(extension, 'custom_js'):
|
|
|
|
all_js += getattr(extension, 'custom_js')()
|
|
|
|
|
|
|
|
return all_js
|
|
|
|
|
|
|
|
|
2023-05-17 00:25:01 -04:00
|
|
|
def create_extensions_block():
|
|
|
|
to_display = []
|
|
|
|
for extension, name in iterator():
|
|
|
|
if hasattr(extension, "ui") and not (hasattr(extension, 'params') and extension.params.get('is_tab', False)):
|
|
|
|
to_display.append((extension, name))
|
|
|
|
|
|
|
|
# Creating the extension ui elements
|
|
|
|
if len(to_display) > 0:
|
|
|
|
with gr.Column(elem_id="extensions"):
|
|
|
|
for row in to_display:
|
|
|
|
extension, name = row
|
|
|
|
display_name = getattr(extension, 'params', {}).get('display_name', name)
|
|
|
|
gr.Markdown(f"\n### {display_name}")
|
|
|
|
extension.ui()
|
|
|
|
|
|
|
|
|
|
|
|
def create_extensions_tabs():
|
|
|
|
for extension, name in iterator():
|
|
|
|
if hasattr(extension, "ui") and (hasattr(extension, 'params') and extension.params.get('is_tab', False)):
|
|
|
|
display_name = getattr(extension, 'params', {}).get('display_name', name)
|
|
|
|
with gr.Tab(display_name, elem_classes="extension-tab"):
|
|
|
|
extension.ui()
|
|
|
|
|
|
|
|
|
2023-04-23 19:32:22 -04:00
|
|
|
EXTENSION_MAP = {
|
|
|
|
"input": partial(_apply_string_extensions, "input_modifier"),
|
|
|
|
"output": partial(_apply_string_extensions, "output_modifier"),
|
2023-05-05 17:53:03 -04:00
|
|
|
"state": _apply_state_modifier_extensions,
|
2023-05-21 12:24:54 -04:00
|
|
|
"history": _apply_history_modifier_extensions,
|
2023-04-23 19:32:22 -04:00
|
|
|
"bot_prefix": partial(_apply_string_extensions, "bot_prefix_modifier"),
|
|
|
|
"tokenizer": partial(_apply_tokenizer_extensions, "tokenizer_modifier"),
|
2023-07-13 16:22:41 -04:00
|
|
|
'logits_processor': partial(_apply_logits_processor_extensions, 'logits_processor_modifier'),
|
2023-04-23 19:32:22 -04:00
|
|
|
"input_hijack": _apply_input_hijack,
|
2023-05-05 17:53:03 -04:00
|
|
|
"custom_generate_chat_prompt": _apply_custom_generate_chat_prompt,
|
2023-05-09 19:18:02 -04:00
|
|
|
"custom_generate_reply": _apply_custom_generate_reply,
|
2023-05-16 23:03:39 -04:00
|
|
|
"tokenized_length": _apply_custom_tokenized_length,
|
|
|
|
"css": _apply_custom_css,
|
|
|
|
"js": _apply_custom_js
|
2023-04-23 19:32:22 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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)
|