mirror of
https://github.com/open-webui/open-webui
synced 2025-01-19 17:26:33 +00:00
300 lines
9.6 KiB
Python
300 lines
9.6 KiB
Python
import logging
|
|
import math
|
|
import re
|
|
from datetime import datetime
|
|
from typing import Optional
|
|
import uuid
|
|
|
|
|
|
from open_webui.utils.misc import get_last_user_message, get_messages_content
|
|
|
|
from open_webui.env import SRC_LOG_LEVELS
|
|
from open_webui.config import DEFAULT_RAG_TEMPLATE
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
log.setLevel(SRC_LOG_LEVELS["RAG"])
|
|
|
|
|
|
def prompt_template(
|
|
template: str, user_name: Optional[str] = None, user_location: Optional[str] = None
|
|
) -> str:
|
|
# Get the current date
|
|
current_date = datetime.now()
|
|
|
|
# Format the date to YYYY-MM-DD
|
|
formatted_date = current_date.strftime("%Y-%m-%d")
|
|
formatted_time = current_date.strftime("%I:%M:%S %p")
|
|
formatted_weekday = current_date.strftime("%A")
|
|
|
|
template = template.replace("{{CURRENT_DATE}}", formatted_date)
|
|
template = template.replace("{{CURRENT_TIME}}", formatted_time)
|
|
template = template.replace(
|
|
"{{CURRENT_DATETIME}}", f"{formatted_date} {formatted_time}"
|
|
)
|
|
template = template.replace("{{CURRENT_WEEKDAY}}", formatted_weekday)
|
|
|
|
if user_name:
|
|
# Replace {{USER_NAME}} in the template with the user's name
|
|
template = template.replace("{{USER_NAME}}", user_name)
|
|
else:
|
|
# Replace {{USER_NAME}} in the template with "Unknown"
|
|
template = template.replace("{{USER_NAME}}", "Unknown")
|
|
|
|
if user_location:
|
|
# Replace {{USER_LOCATION}} in the template with the current location
|
|
template = template.replace("{{USER_LOCATION}}", user_location)
|
|
else:
|
|
# Replace {{USER_LOCATION}} in the template with "Unknown"
|
|
template = template.replace("{{USER_LOCATION}}", "Unknown")
|
|
|
|
return template
|
|
|
|
|
|
def replace_prompt_variable(template: str, prompt: str) -> str:
|
|
def replacement_function(match):
|
|
full_match = match.group(
|
|
0
|
|
).lower() # Normalize to lowercase for consistent handling
|
|
start_length = match.group(1)
|
|
end_length = match.group(2)
|
|
middle_length = match.group(3)
|
|
|
|
if full_match == "{{prompt}}":
|
|
return prompt
|
|
elif start_length is not None:
|
|
return prompt[: int(start_length)]
|
|
elif end_length is not None:
|
|
return prompt[-int(end_length) :]
|
|
elif middle_length is not None:
|
|
middle_length = int(middle_length)
|
|
if len(prompt) <= middle_length:
|
|
return prompt
|
|
start = prompt[: math.ceil(middle_length / 2)]
|
|
end = prompt[-math.floor(middle_length / 2) :]
|
|
return f"{start}...{end}"
|
|
return ""
|
|
|
|
# Updated regex pattern to make it case-insensitive with the `(?i)` flag
|
|
pattern = r"(?i){{prompt}}|{{prompt:start:(\d+)}}|{{prompt:end:(\d+)}}|{{prompt:middletruncate:(\d+)}}"
|
|
template = re.sub(pattern, replacement_function, template)
|
|
return template
|
|
|
|
|
|
def replace_messages_variable(
|
|
template: str, messages: Optional[list[str]] = None
|
|
) -> str:
|
|
def replacement_function(match):
|
|
full_match = match.group(0)
|
|
start_length = match.group(1)
|
|
end_length = match.group(2)
|
|
middle_length = match.group(3)
|
|
# If messages is None, handle it as an empty list
|
|
if messages is None:
|
|
return ""
|
|
|
|
# Process messages based on the number of messages required
|
|
if full_match == "{{MESSAGES}}":
|
|
return get_messages_content(messages)
|
|
elif start_length is not None:
|
|
return get_messages_content(messages[: int(start_length)])
|
|
elif end_length is not None:
|
|
return get_messages_content(messages[-int(end_length) :])
|
|
elif middle_length is not None:
|
|
mid = int(middle_length)
|
|
|
|
if len(messages) <= mid:
|
|
return get_messages_content(messages)
|
|
# Handle middle truncation: split to get start and end portions of the messages list
|
|
half = mid // 2
|
|
start_msgs = messages[:half]
|
|
end_msgs = messages[-half:] if mid % 2 == 0 else messages[-(half + 1) :]
|
|
formatted_start = get_messages_content(start_msgs)
|
|
formatted_end = get_messages_content(end_msgs)
|
|
return f"{formatted_start}\n{formatted_end}"
|
|
return ""
|
|
|
|
template = re.sub(
|
|
r"{{MESSAGES}}|{{MESSAGES:START:(\d+)}}|{{MESSAGES:END:(\d+)}}|{{MESSAGES:MIDDLETRUNCATE:(\d+)}}",
|
|
replacement_function,
|
|
template,
|
|
)
|
|
|
|
return template
|
|
|
|
|
|
# {{prompt:middletruncate:8000}}
|
|
|
|
|
|
def rag_template(template: str, context: str, query: str):
|
|
if template.strip() == "":
|
|
template = DEFAULT_RAG_TEMPLATE
|
|
|
|
if "[context]" not in template and "{{CONTEXT}}" not in template:
|
|
log.debug(
|
|
"WARNING: The RAG template does not contain the '[context]' or '{{CONTEXT}}' placeholder."
|
|
)
|
|
|
|
if "<context>" in context and "</context>" in context:
|
|
log.debug(
|
|
"WARNING: Potential prompt injection attack: the RAG "
|
|
"context contains '<context>' and '</context>'. This might be "
|
|
"nothing, or the user might be trying to hack something."
|
|
)
|
|
|
|
query_placeholders = []
|
|
if "[query]" in context:
|
|
query_placeholder = "{{QUERY" + str(uuid.uuid4()) + "}}"
|
|
template = template.replace("[query]", query_placeholder)
|
|
query_placeholders.append(query_placeholder)
|
|
|
|
if "{{QUERY}}" in context:
|
|
query_placeholder = "{{QUERY" + str(uuid.uuid4()) + "}}"
|
|
template = template.replace("{{QUERY}}", query_placeholder)
|
|
query_placeholders.append(query_placeholder)
|
|
|
|
template = template.replace("[context]", context)
|
|
template = template.replace("{{CONTEXT}}", context)
|
|
template = template.replace("[query]", query)
|
|
template = template.replace("{{QUERY}}", query)
|
|
|
|
for query_placeholder in query_placeholders:
|
|
template = template.replace(query_placeholder, query)
|
|
|
|
return template
|
|
|
|
|
|
def title_generation_template(
|
|
template: str, messages: list[dict], user: Optional[dict] = None
|
|
) -> str:
|
|
prompt = get_last_user_message(messages)
|
|
template = replace_prompt_variable(template, prompt)
|
|
template = replace_messages_variable(template, messages)
|
|
|
|
template = prompt_template(
|
|
template,
|
|
**(
|
|
{"user_name": user.get("name"), "user_location": user.get("location")}
|
|
if user
|
|
else {}
|
|
),
|
|
)
|
|
|
|
return template
|
|
|
|
|
|
def tags_generation_template(
|
|
template: str, messages: list[dict], user: Optional[dict] = None
|
|
) -> str:
|
|
prompt = get_last_user_message(messages)
|
|
template = replace_prompt_variable(template, prompt)
|
|
template = replace_messages_variable(template, messages)
|
|
|
|
template = prompt_template(
|
|
template,
|
|
**(
|
|
{"user_name": user.get("name"), "user_location": user.get("location")}
|
|
if user
|
|
else {}
|
|
),
|
|
)
|
|
return template
|
|
|
|
|
|
def emoji_generation_template(
|
|
template: str, prompt: str, user: Optional[dict] = None
|
|
) -> str:
|
|
template = replace_prompt_variable(template, prompt)
|
|
template = prompt_template(
|
|
template,
|
|
**(
|
|
{"user_name": user.get("name"), "user_location": user.get("location")}
|
|
if user
|
|
else {}
|
|
),
|
|
)
|
|
|
|
return template
|
|
|
|
|
|
def autocomplete_generation_template(
|
|
template: str,
|
|
prompt: str,
|
|
messages: Optional[list[dict]] = None,
|
|
type: Optional[str] = None,
|
|
user: Optional[dict] = None,
|
|
) -> str:
|
|
template = template.replace("{{TYPE}}", type if type else "")
|
|
template = replace_prompt_variable(template, prompt)
|
|
template = replace_messages_variable(template, messages)
|
|
|
|
template = prompt_template(
|
|
template,
|
|
**(
|
|
{"user_name": user.get("name"), "user_location": user.get("location")}
|
|
if user
|
|
else {}
|
|
),
|
|
)
|
|
return template
|
|
|
|
|
|
def query_generation_template(
|
|
template: str, messages: list[dict], user: Optional[dict] = None
|
|
) -> str:
|
|
prompt = get_last_user_message(messages)
|
|
template = replace_prompt_variable(template, prompt)
|
|
template = replace_messages_variable(template, messages)
|
|
|
|
template = prompt_template(
|
|
template,
|
|
**(
|
|
{"user_name": user.get("name"), "user_location": user.get("location")}
|
|
if user
|
|
else {}
|
|
),
|
|
)
|
|
return template
|
|
|
|
|
|
def moa_response_generation_template(
|
|
template: str, prompt: str, responses: list[str]
|
|
) -> str:
|
|
def replacement_function(match):
|
|
full_match = match.group(0)
|
|
start_length = match.group(1)
|
|
end_length = match.group(2)
|
|
middle_length = match.group(3)
|
|
|
|
if full_match == "{{prompt}}":
|
|
return prompt
|
|
elif start_length is not None:
|
|
return prompt[: int(start_length)]
|
|
elif end_length is not None:
|
|
return prompt[-int(end_length) :]
|
|
elif middle_length is not None:
|
|
middle_length = int(middle_length)
|
|
if len(prompt) <= middle_length:
|
|
return prompt
|
|
start = prompt[: math.ceil(middle_length / 2)]
|
|
end = prompt[-math.floor(middle_length / 2) :]
|
|
return f"{start}...{end}"
|
|
return ""
|
|
|
|
template = re.sub(
|
|
r"{{prompt}}|{{prompt:start:(\d+)}}|{{prompt:end:(\d+)}}|{{prompt:middletruncate:(\d+)}}",
|
|
replacement_function,
|
|
template,
|
|
)
|
|
|
|
responses = [f'"""{response}"""' for response in responses]
|
|
responses = "\n\n".join(responses)
|
|
|
|
template = template.replace("{{responses}}", responses)
|
|
return template
|
|
|
|
|
|
def tools_function_calling_generation_template(template: str, tools_specs: str) -> str:
|
|
template = template.replace("{{TOOLS}}", tools_specs)
|
|
return template
|