task.py 7.0 KB

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