task.py 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  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. formatted_weekday = current_date.strftime("%A")
  21. template = template.replace("{{CURRENT_DATE}}", formatted_date)
  22. template = template.replace("{{CURRENT_TIME}}", formatted_time)
  23. template = template.replace(
  24. "{{CURRENT_DATETIME}}", f"{formatted_date} {formatted_time}"
  25. )
  26. template = template.replace("{{CURRENT_WEEKDAY}}", formatted_weekday)
  27. if user_name:
  28. # Replace {{USER_NAME}} in the template with the user's name
  29. template = template.replace("{{USER_NAME}}", user_name)
  30. else:
  31. # Replace {{USER_NAME}} in the template with "Unknown"
  32. template = template.replace("{{USER_NAME}}", "Unknown")
  33. if user_location:
  34. # Replace {{USER_LOCATION}} in the template with the current location
  35. template = template.replace("{{USER_LOCATION}}", user_location)
  36. else:
  37. # Replace {{USER_LOCATION}} in the template with "Unknown"
  38. template = template.replace("{{USER_LOCATION}}", "Unknown")
  39. return template
  40. def replace_prompt_variable(template: str, prompt: str) -> str:
  41. def replacement_function(match):
  42. full_match = match.group(
  43. 0
  44. ).lower() # Normalize to lowercase for consistent handling
  45. start_length = match.group(1)
  46. end_length = match.group(2)
  47. middle_length = match.group(3)
  48. if full_match == "{{prompt}}":
  49. return prompt
  50. elif start_length is not None:
  51. return prompt[: int(start_length)]
  52. elif end_length is not None:
  53. return prompt[-int(end_length) :]
  54. elif middle_length is not None:
  55. middle_length = int(middle_length)
  56. if len(prompt) <= middle_length:
  57. return prompt
  58. start = prompt[: math.ceil(middle_length / 2)]
  59. end = prompt[-math.floor(middle_length / 2) :]
  60. return f"{start}...{end}"
  61. return ""
  62. # Updated regex pattern to make it case-insensitive with the `(?i)` flag
  63. pattern = r"(?i){{prompt}}|{{prompt:start:(\d+)}}|{{prompt:end:(\d+)}}|{{prompt:middletruncate:(\d+)}}"
  64. template = re.sub(pattern, replacement_function, template)
  65. return template
  66. def replace_messages_variable(
  67. template: str, messages: Optional[list[str]] = None
  68. ) -> str:
  69. def replacement_function(match):
  70. full_match = match.group(0)
  71. start_length = match.group(1)
  72. end_length = match.group(2)
  73. middle_length = match.group(3)
  74. # If messages is None, handle it as an empty list
  75. if messages is None:
  76. return ""
  77. # Process messages based on the number of messages required
  78. if full_match == "{{MESSAGES}}":
  79. return get_messages_content(messages)
  80. elif start_length is not None:
  81. return get_messages_content(messages[: int(start_length)])
  82. elif end_length is not None:
  83. return get_messages_content(messages[-int(end_length) :])
  84. elif middle_length is not None:
  85. mid = int(middle_length)
  86. if len(messages) <= mid:
  87. return get_messages_content(messages)
  88. # Handle middle truncation: split to get start and end portions of the messages list
  89. half = mid // 2
  90. start_msgs = messages[:half]
  91. end_msgs = messages[-half:] if mid % 2 == 0 else messages[-(half + 1) :]
  92. formatted_start = get_messages_content(start_msgs)
  93. formatted_end = get_messages_content(end_msgs)
  94. return f"{formatted_start}\n{formatted_end}"
  95. return ""
  96. template = re.sub(
  97. r"{{MESSAGES}}|{{MESSAGES:START:(\d+)}}|{{MESSAGES:END:(\d+)}}|{{MESSAGES:MIDDLETRUNCATE:(\d+)}}",
  98. replacement_function,
  99. template,
  100. )
  101. return template
  102. # {{prompt:middletruncate:8000}}
  103. def rag_template(template: str, context: str, query: str):
  104. if template.strip() == "":
  105. template = DEFAULT_RAG_TEMPLATE
  106. if "[context]" not in template and "{{CONTEXT}}" not in template:
  107. log.debug(
  108. "WARNING: The RAG template does not contain the '[context]' or '{{CONTEXT}}' placeholder."
  109. )
  110. if "<context>" in context and "</context>" in context:
  111. log.debug(
  112. "WARNING: Potential prompt injection attack: the RAG "
  113. "context contains '<context>' and '</context>'. This might be "
  114. "nothing, or the user might be trying to hack something."
  115. )
  116. query_placeholders = []
  117. if "[query]" in context:
  118. query_placeholder = "{{QUERY" + str(uuid.uuid4()) + "}}"
  119. template = template.replace("[query]", query_placeholder)
  120. query_placeholders.append(query_placeholder)
  121. if "{{QUERY}}" in context:
  122. query_placeholder = "{{QUERY" + str(uuid.uuid4()) + "}}"
  123. template = template.replace("{{QUERY}}", query_placeholder)
  124. query_placeholders.append(query_placeholder)
  125. template = template.replace("[context]", context)
  126. template = template.replace("{{CONTEXT}}", context)
  127. template = template.replace("[query]", query)
  128. template = template.replace("{{QUERY}}", query)
  129. for query_placeholder in query_placeholders:
  130. template = template.replace(query_placeholder, query)
  131. return template
  132. def title_generation_template(
  133. template: str, messages: list[dict], user: Optional[dict] = None
  134. ) -> str:
  135. prompt = get_last_user_message(messages)
  136. template = replace_prompt_variable(template, prompt)
  137. template = replace_messages_variable(template, messages)
  138. template = prompt_template(
  139. template,
  140. **(
  141. {"user_name": user.get("name"), "user_location": user.get("location")}
  142. if user
  143. else {}
  144. ),
  145. )
  146. return template
  147. def tags_generation_template(
  148. template: str, messages: list[dict], user: Optional[dict] = None
  149. ) -> str:
  150. prompt = get_last_user_message(messages)
  151. template = replace_prompt_variable(template, prompt)
  152. template = replace_messages_variable(template, messages)
  153. template = prompt_template(
  154. template,
  155. **(
  156. {"user_name": user.get("name"), "user_location": user.get("location")}
  157. if user
  158. else {}
  159. ),
  160. )
  161. return template
  162. def emoji_generation_template(
  163. template: str, prompt: str, user: Optional[dict] = None
  164. ) -> str:
  165. template = replace_prompt_variable(template, prompt)
  166. template = prompt_template(
  167. template,
  168. **(
  169. {"user_name": user.get("name"), "user_location": user.get("location")}
  170. if user
  171. else {}
  172. ),
  173. )
  174. return template
  175. def autocomplete_generation_template(
  176. template: str,
  177. prompt: str,
  178. messages: Optional[list[dict]] = None,
  179. type: Optional[str] = None,
  180. user: Optional[dict] = None,
  181. ) -> str:
  182. template = template.replace("{{TYPE}}", type if type else "")
  183. template = replace_prompt_variable(template, prompt)
  184. template = replace_messages_variable(template, messages)
  185. template = prompt_template(
  186. template,
  187. **(
  188. {"user_name": user.get("name"), "user_location": user.get("location")}
  189. if user
  190. else {}
  191. ),
  192. )
  193. return template
  194. def query_generation_template(
  195. template: str, messages: list[dict], user: Optional[dict] = None
  196. ) -> str:
  197. prompt = get_last_user_message(messages)
  198. template = replace_prompt_variable(template, prompt)
  199. template = replace_messages_variable(template, messages)
  200. template = prompt_template(
  201. template,
  202. **(
  203. {"user_name": user.get("name"), "user_location": user.get("location")}
  204. if user
  205. else {}
  206. ),
  207. )
  208. return template
  209. def moa_response_generation_template(
  210. template: str, prompt: str, responses: list[str]
  211. ) -> str:
  212. def replacement_function(match):
  213. full_match = match.group(0)
  214. start_length = match.group(1)
  215. end_length = match.group(2)
  216. middle_length = match.group(3)
  217. if full_match == "{{prompt}}":
  218. return prompt
  219. elif start_length is not None:
  220. return prompt[: int(start_length)]
  221. elif end_length is not None:
  222. return prompt[-int(end_length) :]
  223. elif middle_length is not None:
  224. middle_length = int(middle_length)
  225. if len(prompt) <= middle_length:
  226. return prompt
  227. start = prompt[: math.ceil(middle_length / 2)]
  228. end = prompt[-math.floor(middle_length / 2) :]
  229. return f"{start}...{end}"
  230. return ""
  231. template = re.sub(
  232. r"{{prompt}}|{{prompt:start:(\d+)}}|{{prompt:end:(\d+)}}|{{prompt:middletruncate:(\d+)}}",
  233. replacement_function,
  234. template,
  235. )
  236. responses = [f'"""{response}"""' for response in responses]
  237. responses = "\n\n".join(responses)
  238. template = template.replace("{{responses}}", responses)
  239. return template
  240. def tools_function_calling_generation_template(template: str, tools_specs: str) -> str:
  241. template = template.replace("{{TOOLS}}", tools_specs)
  242. return template