completion_service.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519
  1. import json
  2. import logging
  3. import threading
  4. import time
  5. import uuid
  6. from typing import Generator, Union, Any
  7. from flask import current_app, Flask
  8. from redis.client import PubSub
  9. from sqlalchemy import and_
  10. from core.completion import Completion
  11. from core.conversation_message_task import PubHandler, ConversationTaskStoppedException
  12. from core.llm.error import LLMBadRequestError, LLMAPIConnectionError, LLMAPIUnavailableError, LLMRateLimitError, \
  13. LLMAuthorizationError, ProviderTokenNotInitError, QuotaExceededError, ModelCurrentlyNotSupportError
  14. from extensions.ext_database import db
  15. from extensions.ext_redis import redis_client
  16. from models.model import Conversation, AppModelConfig, App, Account, EndUser, Message
  17. from services.app_model_config_service import AppModelConfigService
  18. from services.errors.app import MoreLikeThisDisabledError
  19. from services.errors.app_model_config import AppModelConfigBrokenError
  20. from services.errors.completion import CompletionStoppedError
  21. from services.errors.conversation import ConversationNotExistsError, ConversationCompletedError
  22. from services.errors.message import MessageNotExistsError
  23. class CompletionService:
  24. @classmethod
  25. def completion(cls, app_model: App, user: Union[Account | EndUser], args: Any,
  26. from_source: str, streaming: bool = True,
  27. is_model_config_override: bool = False) -> Union[dict | Generator]:
  28. # is streaming mode
  29. inputs = args['inputs']
  30. query = args['query']
  31. if not query:
  32. raise ValueError('query is required')
  33. query = query.replace('\x00', '')
  34. conversation_id = args['conversation_id'] if 'conversation_id' in args else None
  35. conversation = None
  36. if conversation_id:
  37. conversation_filter = [
  38. Conversation.id == args['conversation_id'],
  39. Conversation.app_id == app_model.id,
  40. Conversation.status == 'normal'
  41. ]
  42. if from_source == 'console':
  43. conversation_filter.append(Conversation.from_account_id == user.id)
  44. else:
  45. conversation_filter.append(Conversation.from_end_user_id == user.id if user else None)
  46. conversation = db.session.query(Conversation).filter(and_(*conversation_filter)).first()
  47. if not conversation:
  48. raise ConversationNotExistsError()
  49. if conversation.status != 'normal':
  50. raise ConversationCompletedError()
  51. if not conversation.override_model_configs:
  52. app_model_config = db.session.query(AppModelConfig).get(conversation.app_model_config_id)
  53. if not app_model_config:
  54. raise AppModelConfigBrokenError()
  55. else:
  56. conversation_override_model_configs = json.loads(conversation.override_model_configs)
  57. app_model_config = AppModelConfig(
  58. id=conversation.app_model_config_id,
  59. app_id=app_model.id,
  60. provider="",
  61. model_id="",
  62. configs="",
  63. opening_statement=conversation_override_model_configs['opening_statement'],
  64. suggested_questions=json.dumps(conversation_override_model_configs['suggested_questions']),
  65. model=json.dumps(conversation_override_model_configs['model']),
  66. user_input_form=json.dumps(conversation_override_model_configs['user_input_form']),
  67. pre_prompt=conversation_override_model_configs['pre_prompt'],
  68. agent_mode=json.dumps(conversation_override_model_configs['agent_mode']),
  69. )
  70. if is_model_config_override:
  71. # build new app model config
  72. if 'model' not in args['model_config']:
  73. raise ValueError('model_config.model is required')
  74. if 'completion_params' not in args['model_config']['model']:
  75. raise ValueError('model_config.model.completion_params is required')
  76. completion_params = AppModelConfigService.validate_model_completion_params(
  77. cp=args['model_config']['model']['completion_params'],
  78. model_name=app_model_config.model_dict["name"]
  79. )
  80. app_model_config_model = app_model_config.model_dict
  81. app_model_config_model['completion_params'] = completion_params
  82. app_model_config = AppModelConfig(
  83. id=app_model_config.id,
  84. app_id=app_model.id,
  85. provider="",
  86. model_id="",
  87. configs="",
  88. opening_statement=app_model_config.opening_statement,
  89. suggested_questions=app_model_config.suggested_questions,
  90. model=json.dumps(app_model_config_model),
  91. user_input_form=app_model_config.user_input_form,
  92. pre_prompt=app_model_config.pre_prompt,
  93. agent_mode=app_model_config.agent_mode,
  94. )
  95. else:
  96. if app_model.app_model_config_id is None:
  97. raise AppModelConfigBrokenError()
  98. app_model_config = app_model.app_model_config
  99. if not app_model_config:
  100. raise AppModelConfigBrokenError()
  101. if is_model_config_override:
  102. if not isinstance(user, Account):
  103. raise Exception("Only account can override model config")
  104. # validate config
  105. model_config = AppModelConfigService.validate_configuration(
  106. account=user,
  107. config=args['model_config'],
  108. mode=app_model.mode
  109. )
  110. app_model_config = AppModelConfig(
  111. id=app_model_config.id,
  112. app_id=app_model.id,
  113. provider="",
  114. model_id="",
  115. configs="",
  116. opening_statement=model_config['opening_statement'],
  117. suggested_questions=json.dumps(model_config['suggested_questions']),
  118. suggested_questions_after_answer=json.dumps(model_config['suggested_questions_after_answer']),
  119. more_like_this=json.dumps(model_config['more_like_this']),
  120. sensitive_word_avoidance=json.dumps(model_config['sensitive_word_avoidance']),
  121. model=json.dumps(model_config['model']),
  122. user_input_form=json.dumps(model_config['user_input_form']),
  123. pre_prompt=model_config['pre_prompt'],
  124. agent_mode=json.dumps(model_config['agent_mode']),
  125. )
  126. # clean input by app_model_config form rules
  127. inputs = cls.get_cleaned_inputs(inputs, app_model_config)
  128. generate_task_id = str(uuid.uuid4())
  129. pubsub = redis_client.pubsub()
  130. pubsub.subscribe(PubHandler.generate_channel_name(user, generate_task_id))
  131. user = cls.get_real_user_instead_of_proxy_obj(user)
  132. generate_worker_thread = threading.Thread(target=cls.generate_worker, kwargs={
  133. 'flask_app': current_app._get_current_object(),
  134. 'generate_task_id': generate_task_id,
  135. 'app_model': app_model,
  136. 'app_model_config': app_model_config,
  137. 'query': query,
  138. 'inputs': inputs,
  139. 'user': user,
  140. 'conversation': conversation,
  141. 'streaming': streaming,
  142. 'is_model_config_override': is_model_config_override
  143. })
  144. generate_worker_thread.start()
  145. # wait for 10 minutes to close the thread
  146. cls.countdown_and_close(generate_worker_thread, pubsub, user, generate_task_id)
  147. return cls.compact_response(pubsub, streaming)
  148. @classmethod
  149. def get_real_user_instead_of_proxy_obj(cls, user: Union[Account, EndUser]):
  150. if isinstance(user, Account):
  151. user = db.session.query(Account).filter(Account.id == user.id).first()
  152. elif isinstance(user, EndUser):
  153. user = db.session.query(EndUser).filter(EndUser.id == user.id).first()
  154. else:
  155. raise Exception("Unknown user type")
  156. return user
  157. @classmethod
  158. def generate_worker(cls, flask_app: Flask, generate_task_id: str, app_model: App, app_model_config: AppModelConfig,
  159. query: str, inputs: dict, user: Union[Account, EndUser],
  160. conversation: Conversation, streaming: bool, is_model_config_override: bool):
  161. with flask_app.app_context():
  162. try:
  163. if conversation:
  164. # fixed the state of the conversation object when it detached from the original session
  165. conversation = db.session.query(Conversation).filter_by(id=conversation.id).first()
  166. # run
  167. Completion.generate(
  168. task_id=generate_task_id,
  169. app=app_model,
  170. app_model_config=app_model_config,
  171. query=query,
  172. inputs=inputs,
  173. user=user,
  174. conversation=conversation,
  175. streaming=streaming,
  176. is_override=is_model_config_override,
  177. )
  178. except ConversationTaskStoppedException:
  179. pass
  180. except (LLMBadRequestError, LLMAPIConnectionError, LLMAPIUnavailableError,
  181. LLMRateLimitError, ProviderTokenNotInitError, QuotaExceededError,
  182. ModelCurrentlyNotSupportError) as e:
  183. db.session.rollback()
  184. PubHandler.pub_error(user, generate_task_id, e)
  185. except LLMAuthorizationError:
  186. db.session.rollback()
  187. PubHandler.pub_error(user, generate_task_id, LLMAuthorizationError('Incorrect API key provided'))
  188. except Exception as e:
  189. db.session.rollback()
  190. logging.exception("Unknown Error in completion")
  191. PubHandler.pub_error(user, generate_task_id, e)
  192. @classmethod
  193. def countdown_and_close(cls, worker_thread, pubsub, user, generate_task_id) -> threading.Thread:
  194. # wait for 10 minutes to close the thread
  195. timeout = 600
  196. def close_pubsub():
  197. sleep_iterations = 0
  198. while sleep_iterations < timeout and worker_thread.is_alive():
  199. if sleep_iterations > 0 and sleep_iterations % 10 == 0:
  200. PubHandler.ping(user, generate_task_id)
  201. time.sleep(1)
  202. sleep_iterations += 1
  203. if worker_thread.is_alive():
  204. PubHandler.stop(user, generate_task_id)
  205. try:
  206. pubsub.close()
  207. except:
  208. pass
  209. countdown_thread = threading.Thread(target=close_pubsub)
  210. countdown_thread.start()
  211. return countdown_thread
  212. @classmethod
  213. def generate_more_like_this(cls, app_model: App, user: Union[Account | EndUser],
  214. message_id: str, streaming: bool = True) -> Union[dict | Generator]:
  215. if not user:
  216. raise ValueError('user cannot be None')
  217. message = db.session.query(Message).filter(
  218. Message.id == message_id,
  219. Message.app_id == app_model.id,
  220. Message.from_source == ('api' if isinstance(user, EndUser) else 'console'),
  221. Message.from_end_user_id == (user.id if isinstance(user, EndUser) else None),
  222. Message.from_account_id == (user.id if isinstance(user, Account) else None),
  223. ).first()
  224. if not message:
  225. raise MessageNotExistsError()
  226. current_app_model_config = app_model.app_model_config
  227. more_like_this = current_app_model_config.more_like_this_dict
  228. if not current_app_model_config.more_like_this or more_like_this.get("enabled", False) is False:
  229. raise MoreLikeThisDisabledError()
  230. app_model_config = message.app_model_config
  231. if message.override_model_configs:
  232. override_model_configs = json.loads(message.override_model_configs)
  233. pre_prompt = override_model_configs.get("pre_prompt", '')
  234. elif app_model_config:
  235. pre_prompt = app_model_config.pre_prompt
  236. else:
  237. raise AppModelConfigBrokenError()
  238. generate_task_id = str(uuid.uuid4())
  239. pubsub = redis_client.pubsub()
  240. pubsub.subscribe(PubHandler.generate_channel_name(user, generate_task_id))
  241. user = cls.get_real_user_instead_of_proxy_obj(user)
  242. generate_worker_thread = threading.Thread(target=cls.generate_more_like_this_worker, kwargs={
  243. 'flask_app': current_app._get_current_object(),
  244. 'generate_task_id': generate_task_id,
  245. 'app_model': app_model,
  246. 'app_model_config': app_model_config,
  247. 'message': message,
  248. 'pre_prompt': pre_prompt,
  249. 'user': user,
  250. 'streaming': streaming
  251. })
  252. generate_worker_thread.start()
  253. cls.countdown_and_close(generate_worker_thread, pubsub, user, generate_task_id)
  254. return cls.compact_response(pubsub, streaming)
  255. @classmethod
  256. def generate_more_like_this_worker(cls, flask_app: Flask, generate_task_id: str, app_model: App,
  257. app_model_config: AppModelConfig, message: Message, pre_prompt: str,
  258. user: Union[Account, EndUser], streaming: bool):
  259. with flask_app.app_context():
  260. try:
  261. # run
  262. Completion.generate_more_like_this(
  263. task_id=generate_task_id,
  264. app=app_model,
  265. user=user,
  266. message=message,
  267. pre_prompt=pre_prompt,
  268. app_model_config=app_model_config,
  269. streaming=streaming
  270. )
  271. except ConversationTaskStoppedException:
  272. pass
  273. except (LLMBadRequestError, LLMAPIConnectionError, LLMAPIUnavailableError,
  274. LLMRateLimitError, ProviderTokenNotInitError, QuotaExceededError,
  275. ModelCurrentlyNotSupportError) as e:
  276. db.session.rollback()
  277. PubHandler.pub_error(user, generate_task_id, e)
  278. except LLMAuthorizationError:
  279. db.session.rollback()
  280. PubHandler.pub_error(user, generate_task_id, LLMAuthorizationError('Incorrect API key provided'))
  281. except Exception as e:
  282. db.session.rollback()
  283. logging.exception("Unknown Error in completion")
  284. PubHandler.pub_error(user, generate_task_id, e)
  285. @classmethod
  286. def get_cleaned_inputs(cls, user_inputs: dict, app_model_config: AppModelConfig):
  287. if user_inputs is None:
  288. user_inputs = {}
  289. filtered_inputs = {}
  290. # Filter input variables from form configuration, handle required fields, default values, and option values
  291. input_form_config = app_model_config.user_input_form_list
  292. for config in input_form_config:
  293. input_config = list(config.values())[0]
  294. variable = input_config["variable"]
  295. input_type = list(config.keys())[0]
  296. if variable not in user_inputs or not user_inputs[variable]:
  297. if "required" in input_config and input_config["required"]:
  298. raise ValueError(f"{variable} is required in input form")
  299. else:
  300. filtered_inputs[variable] = input_config["default"] if "default" in input_config else ""
  301. continue
  302. value = user_inputs[variable]
  303. if input_type == "select":
  304. options = input_config["options"] if "options" in input_config else []
  305. if value not in options:
  306. raise ValueError(f"{variable} in input form must be one of the following: {options}")
  307. else:
  308. if 'max_length' in variable:
  309. max_length = variable['max_length']
  310. if len(value) > max_length:
  311. raise ValueError(f'{variable} in input form must be less than {max_length} characters')
  312. filtered_inputs[variable] = value.replace('\x00', '') if value else None
  313. return filtered_inputs
  314. @classmethod
  315. def compact_response(cls, pubsub: PubSub, streaming: bool = False) -> Union[dict | Generator]:
  316. generate_channel = list(pubsub.channels.keys())[0].decode('utf-8')
  317. if not streaming:
  318. try:
  319. for message in pubsub.listen():
  320. if message["type"] == "message":
  321. result = message["data"].decode('utf-8')
  322. result = json.loads(result)
  323. if result.get('error'):
  324. cls.handle_error(result)
  325. if 'data' in result:
  326. return cls.get_message_response_data(result.get('data'))
  327. except ValueError as e:
  328. if e.args[0] != "I/O operation on closed file.": # ignore this error
  329. raise CompletionStoppedError()
  330. else:
  331. logging.exception(e)
  332. raise
  333. finally:
  334. try:
  335. pubsub.unsubscribe(generate_channel)
  336. except ConnectionError:
  337. pass
  338. else:
  339. def generate() -> Generator:
  340. try:
  341. for message in pubsub.listen():
  342. if message["type"] == "message":
  343. result = message["data"].decode('utf-8')
  344. result = json.loads(result)
  345. if result.get('error'):
  346. cls.handle_error(result)
  347. event = result.get('event')
  348. if event == "end":
  349. logging.debug("{} finished".format(generate_channel))
  350. break
  351. if event == 'message':
  352. yield "data: " + json.dumps(cls.get_message_response_data(result.get('data'))) + "\n\n"
  353. elif event == 'chain':
  354. yield "data: " + json.dumps(cls.get_chain_response_data(result.get('data'))) + "\n\n"
  355. elif event == 'agent_thought':
  356. yield "data: " + json.dumps(cls.get_agent_thought_response_data(result.get('data'))) + "\n\n"
  357. elif event == 'ping':
  358. yield "event: ping\n\n"
  359. else:
  360. yield "data: " + json.dumps(result) + "\n\n"
  361. except ValueError as e:
  362. if e.args[0] != "I/O operation on closed file.": # ignore this error
  363. logging.exception(e)
  364. raise
  365. finally:
  366. try:
  367. pubsub.unsubscribe(generate_channel)
  368. except ConnectionError:
  369. pass
  370. return generate()
  371. @classmethod
  372. def get_message_response_data(cls, data: dict):
  373. response_data = {
  374. 'event': 'message',
  375. 'task_id': data.get('task_id'),
  376. 'id': data.get('message_id'),
  377. 'answer': data.get('text'),
  378. 'created_at': int(time.time())
  379. }
  380. if data.get('mode') == 'chat':
  381. response_data['conversation_id'] = data.get('conversation_id')
  382. return response_data
  383. @classmethod
  384. def get_chain_response_data(cls, data: dict):
  385. response_data = {
  386. 'event': 'chain',
  387. 'id': data.get('chain_id'),
  388. 'task_id': data.get('task_id'),
  389. 'message_id': data.get('message_id'),
  390. 'type': data.get('type'),
  391. 'input': data.get('input'),
  392. 'output': data.get('output'),
  393. 'created_at': int(time.time())
  394. }
  395. if data.get('mode') == 'chat':
  396. response_data['conversation_id'] = data.get('conversation_id')
  397. return response_data
  398. @classmethod
  399. def get_agent_thought_response_data(cls, data: dict):
  400. response_data = {
  401. 'event': 'agent_thought',
  402. 'id': data.get('id'),
  403. 'chain_id': data.get('chain_id'),
  404. 'task_id': data.get('task_id'),
  405. 'message_id': data.get('message_id'),
  406. 'position': data.get('position'),
  407. 'thought': data.get('thought'),
  408. 'tool': data.get('tool'),
  409. 'tool_input': data.get('tool_input'),
  410. 'created_at': int(time.time())
  411. }
  412. if data.get('mode') == 'chat':
  413. response_data['conversation_id'] = data.get('conversation_id')
  414. return response_data
  415. @classmethod
  416. def handle_error(cls, result: dict):
  417. logging.debug("error: %s", result)
  418. error = result.get('error')
  419. description = result.get('description')
  420. # handle errors
  421. llm_errors = {
  422. 'LLMBadRequestError': LLMBadRequestError,
  423. 'LLMAPIConnectionError': LLMAPIConnectionError,
  424. 'LLMAPIUnavailableError': LLMAPIUnavailableError,
  425. 'LLMRateLimitError': LLMRateLimitError,
  426. 'ProviderTokenNotInitError': ProviderTokenNotInitError,
  427. 'QuotaExceededError': QuotaExceededError,
  428. 'ModelCurrentlyNotSupportError': ModelCurrentlyNotSupportError
  429. }
  430. if error in llm_errors:
  431. raise llm_errors[error](description)
  432. elif error == 'LLMAuthorizationError':
  433. raise LLMAuthorizationError('Incorrect API key provided')
  434. else:
  435. raise Exception(description)