task.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. import logging
  2. import math
  3. import re
  4. from datetime import datetime
  5. from typing import Optional
  6. import uuid
  7. from open_webui.utils.misc import get_last_user_message, get_messages_content
  8. from open_webui.env import SRC_LOG_LEVELS
  9. from open_webui.config import DEFAULT_RAG_TEMPLATE
  10. log = logging.getLogger(__name__)
  11. log.setLevel(SRC_LOG_LEVELS["RAG"])
  12. def prompt_template(
  13. template: str, user_name: Optional[str] = None, user_location: Optional[str] = None
  14. ) -> str:
  15. # Get the current date
  16. current_date = datetime.now()
  17. # Format the date to YYYY-MM-DD
  18. formatted_date = current_date.strftime("%Y-%m-%d")
  19. formatted_time = current_date.strftime("%I:%M:%S %p")
  20. template = template.replace("{{CURRENT_DATE}}", formatted_date)
  21. template = template.replace("{{CURRENT_TIME}}", formatted_time)
  22. template = template.replace(
  23. "{{CURRENT_DATETIME}}", f"{formatted_date} {formatted_time}"
  24. )
  25. if user_name:
  26. # Replace {{USER_NAME}} in the template with the user's name
  27. template = template.replace("{{USER_NAME}}", user_name)
  28. else:
  29. # Replace {{USER_NAME}} in the template with "Unknown"
  30. template = template.replace("{{USER_NAME}}", "Unknown")
  31. if user_location:
  32. # Replace {{USER_LOCATION}} in the template with the current location
  33. template = template.replace("{{USER_LOCATION}}", user_location)
  34. else:
  35. # Replace {{USER_LOCATION}} in the template with "Unknown"
  36. template = template.replace("{{USER_LOCATION}}", "Unknown")
  37. return template
  38. def replace_prompt_variable(template: str, prompt: str) -> str:
  39. def replacement_function(match):
  40. full_match = match.group(0)
  41. start_length = match.group(1)
  42. end_length = match.group(2)
  43. middle_length = match.group(3)
  44. if full_match == "{{prompt}}":
  45. return prompt
  46. elif start_length is not None:
  47. return prompt[: int(start_length)]
  48. elif end_length is not None:
  49. return prompt[-int(end_length) :]
  50. elif middle_length is not None:
  51. middle_length = int(middle_length)
  52. if len(prompt) <= middle_length:
  53. return prompt
  54. start = prompt[: math.ceil(middle_length / 2)]
  55. end = prompt[-math.floor(middle_length / 2) :]
  56. return f"{start}...{end}"
  57. return ""
  58. template = re.sub(
  59. r"{{prompt}}|{{prompt:start:(\d+)}}|{{prompt:end:(\d+)}}|{{prompt:middletruncate:(\d+)}}",
  60. replacement_function,
  61. template,
  62. )
  63. return template
  64. def replace_messages_variable(template: str, messages: list[str]) -> str:
  65. def replacement_function(match):
  66. full_match = match.group(0)
  67. start_length = match.group(1)
  68. end_length = match.group(2)
  69. middle_length = match.group(3)
  70. # Process messages based on the number of messages required
  71. if full_match == "{{MESSAGES}}":
  72. return get_messages_content(messages)
  73. elif start_length is not None:
  74. return get_messages_content(messages[: int(start_length)])
  75. elif end_length is not None:
  76. return get_messages_content(messages[-int(end_length) :])
  77. elif middle_length is not None:
  78. mid = int(middle_length)
  79. if len(messages) <= mid:
  80. return get_messages_content(messages)
  81. # Handle middle truncation: split to get start and end portions of the messages list
  82. half = mid // 2
  83. start_msgs = messages[:half]
  84. end_msgs = messages[-half:] if mid % 2 == 0 else messages[-(half + 1) :]
  85. formatted_start = get_messages_content(start_msgs)
  86. formatted_end = get_messages_content(end_msgs)
  87. return f"{formatted_start}\n{formatted_end}"
  88. return ""
  89. template = re.sub(
  90. r"{{MESSAGES}}|{{MESSAGES:START:(\d+)}}|{{MESSAGES:END:(\d+)}}|{{MESSAGES:MIDDLETRUNCATE:(\d+)}}",
  91. replacement_function,
  92. template,
  93. )
  94. return template
  95. # {{prompt:middletruncate:8000}}
  96. def rag_template(template: str, context: str, query: str):
  97. if template == "":
  98. template = DEFAULT_RAG_TEMPLATE
  99. if "[context]" not in template and "{{CONTEXT}}" not in template:
  100. log.debug(
  101. "WARNING: The RAG template does not contain the '[context]' or '{{CONTEXT}}' placeholder."
  102. )
  103. if "<context>" in context and "</context>" in context:
  104. log.debug(
  105. "WARNING: Potential prompt injection attack: the RAG "
  106. "context contains '<context>' and '</context>'. This might be "
  107. "nothing, or the user might be trying to hack something."
  108. )
  109. query_placeholders = []
  110. if "[query]" in context:
  111. query_placeholder = "{{QUERY" + str(uuid.uuid4()) + "}}"
  112. template = template.replace("[query]", query_placeholder)
  113. query_placeholders.append(query_placeholder)
  114. if "{{QUERY}}" in context:
  115. query_placeholder = "{{QUERY" + str(uuid.uuid4()) + "}}"
  116. template = template.replace("{{QUERY}}", query_placeholder)
  117. query_placeholders.append(query_placeholder)
  118. template = template.replace("[context]", context)
  119. template = template.replace("{{CONTEXT}}", context)
  120. template = template.replace("[query]", query)
  121. template = template.replace("{{QUERY}}", query)
  122. for query_placeholder in query_placeholders:
  123. template = template.replace(query_placeholder, query)
  124. return template
  125. def title_generation_template(
  126. template: str, messages: list[dict], user: Optional[dict] = None
  127. ) -> str:
  128. prompt = get_last_user_message(messages)
  129. template = replace_prompt_variable(template, prompt)
  130. template = replace_messages_variable(template, messages)
  131. template = prompt_template(
  132. template,
  133. **(
  134. {"user_name": user.get("name"), "user_location": user.get("location")}
  135. if user
  136. else {}
  137. ),
  138. )
  139. return template
  140. def tags_generation_template(
  141. template: str, messages: list[dict], user: Optional[dict] = None
  142. ) -> str:
  143. prompt = get_last_user_message(messages)
  144. template = replace_prompt_variable(template, prompt)
  145. template = replace_messages_variable(template, messages)
  146. template = prompt_template(
  147. template,
  148. **(
  149. {"user_name": user.get("name"), "user_location": user.get("location")}
  150. if user
  151. else {}
  152. ),
  153. )
  154. return template
  155. def emoji_generation_template(
  156. template: str, prompt: str, user: Optional[dict] = None
  157. ) -> str:
  158. template = replace_prompt_variable(template, prompt)
  159. template = prompt_template(
  160. template,
  161. **(
  162. {"user_name": user.get("name"), "user_location": user.get("location")}
  163. if user
  164. else {}
  165. ),
  166. )
  167. return template
  168. def query_generation_template(
  169. template: str, messages: list[dict], user: Optional[dict] = None
  170. ) -> str:
  171. prompt = get_last_user_message(messages)
  172. template = replace_prompt_variable(template, prompt)
  173. template = replace_messages_variable(template, messages)
  174. template = prompt_template(
  175. template,
  176. **(
  177. {"user_name": user.get("name"), "user_location": user.get("location")}
  178. if user
  179. else {}
  180. ),
  181. )
  182. return template
  183. def moa_response_generation_template(
  184. template: str, prompt: str, responses: list[str]
  185. ) -> str:
  186. def replacement_function(match):
  187. full_match = match.group(0)
  188. start_length = match.group(1)
  189. end_length = match.group(2)
  190. middle_length = match.group(3)
  191. if full_match == "{{prompt}}":
  192. return prompt
  193. elif start_length is not None:
  194. return prompt[: int(start_length)]
  195. elif end_length is not None:
  196. return prompt[-int(end_length) :]
  197. elif middle_length is not None:
  198. middle_length = int(middle_length)
  199. if len(prompt) <= middle_length:
  200. return prompt
  201. start = prompt[: math.ceil(middle_length / 2)]
  202. end = prompt[-math.floor(middle_length / 2) :]
  203. return f"{start}...{end}"
  204. return ""
  205. template = re.sub(
  206. r"{{prompt}}|{{prompt:start:(\d+)}}|{{prompt:end:(\d+)}}|{{prompt:middletruncate:(\d+)}}",
  207. replacement_function,
  208. template,
  209. )
  210. responses = [f'"""{response}"""' for response in responses]
  211. responses = "\n\n".join(responses)
  212. template = template.replace("{{responses}}", responses)
  213. return template
  214. def tools_function_calling_generation_template(template: str, tools_specs: str) -> str:
  215. template = template.replace("{{TOOLS}}", tools_specs)
  216. return template