task.py 11 KB

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