| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | module DiscourseAi | 
					
						
							|  |  |  |   module Automation | 
					
						
							|  |  |  |     module LlmTriage | 
					
						
							|  |  |  |       def self.handle( | 
					
						
							|  |  |  |         post:, | 
					
						
							|  |  |  |         model:, | 
					
						
							|  |  |  |         search_for_text:, | 
					
						
							|  |  |  |         system_prompt:, | 
					
						
							|  |  |  |         category_id: nil, | 
					
						
							|  |  |  |         tags: nil, | 
					
						
							|  |  |  |         canned_reply: nil, | 
					
						
							|  |  |  |         canned_reply_user: nil, | 
					
						
							| 
									
										
										
										
											2024-02-28 22:33:28 -03:00
										 |  |  |         hide_topic: nil, | 
					
						
							| 
									
										
										
										
											2024-08-14 15:54:09 -03:00
										 |  |  |         flag_post: nil, | 
					
						
							| 
									
										
										
										
											2024-10-04 15:11:30 +09:00
										 |  |  |         flag_type: nil, | 
					
						
							| 
									
										
										
										
											2024-10-23 16:49:56 +11:00
										 |  |  |         automation: nil, | 
					
						
							| 
									
										
										
										
											2024-12-06 11:13:47 -03:00
										 |  |  |         max_post_tokens: nil, | 
					
						
							| 
									
										
										
										
											2025-03-04 12:22:30 +11:00
										 |  |  |         stop_sequences: nil, | 
					
						
							| 
									
										
										
										
											2025-03-06 17:18:15 +11:00
										 |  |  |         temperature: nil, | 
					
						
							|  |  |  |         whisper: nil, | 
					
						
							|  |  |  |         reply_persona_id: nil, | 
					
						
							|  |  |  |         action: nil | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |       ) | 
					
						
							| 
									
										
										
										
											2024-02-28 22:33:28 -03:00
										 |  |  |         if category_id.blank? && tags.blank? && canned_reply.blank? && hide_topic.blank? && | 
					
						
							| 
									
										
										
										
											2025-03-06 17:18:15 +11:00
										 |  |  |              flag_post.blank? && reply_persona_id.blank? | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |           raise ArgumentError, "llm_triage: no action specified!" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-06 17:18:15 +11:00
										 |  |  |         if action == :edit && category_id.blank? && tags.blank? && flag_post.blank? && | 
					
						
							|  |  |  |              hide_topic.blank? | 
					
						
							|  |  |  |           return | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-23 16:49:56 +11:00
										 |  |  |         llm = DiscourseAi::Completions::Llm.proxy(model) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-21 11:41:55 -03:00
										 |  |  |         s_prompt = system_prompt.to_s.sub("%%POST%%", "") # Backwards-compat. We no longer sub this. | 
					
						
							|  |  |  |         prompt = DiscourseAi::Completions::Prompt.new(s_prompt) | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-23 16:49:56 +11:00
										 |  |  |         content = "title: #{post.topic.title}\n#{post.raw}" | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-23 16:49:56 +11:00
										 |  |  |         content = llm.tokenizer.truncate(content, max_post_tokens) if max_post_tokens.present? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-01 02:39:07 +11:00
										 |  |  |         if post.upload_ids.present? | 
					
						
							|  |  |  |           content = [content] | 
					
						
							|  |  |  |           content.concat(post.upload_ids.map { |upload_id| { upload_id: upload_id } }) | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         prompt.push(type: :user, content: content) | 
					
						
							| 
									
										
										
										
											2024-10-23 16:49:56 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |         result = nil | 
					
						
							| 
									
										
										
										
											2023-12-20 07:58:38 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-04 23:53:47 +11:00
										 |  |  |         result = | 
					
						
							|  |  |  |           llm.generate( | 
					
						
							| 
									
										
										
										
											2024-08-21 11:41:55 -03:00
										 |  |  |             prompt, | 
					
						
							| 
									
										
										
										
											2025-03-04 12:22:30 +11:00
										 |  |  |             temperature: temperature, | 
					
						
							| 
									
										
										
										
											2024-08-14 15:54:09 -03:00
										 |  |  |             max_tokens: 700, # ~500 words | 
					
						
							| 
									
										
										
										
											2024-01-04 23:53:47 +11:00
										 |  |  |             user: Discourse.system_user, | 
					
						
							| 
									
										
										
										
											2024-12-06 11:13:47 -03:00
										 |  |  |             stop_sequences: stop_sequences, | 
					
						
							| 
									
										
										
										
											2024-05-14 13:28:46 +10:00
										 |  |  |             feature_name: "llm_triage", | 
					
						
							| 
									
										
										
										
											2024-10-23 16:49:56 +11:00
										 |  |  |             feature_context: { | 
					
						
							|  |  |  |               automation_id: automation&.id, | 
					
						
							|  |  |  |               automation_name: automation&.name, | 
					
						
							|  |  |  |             }, | 
					
						
							| 
									
										
										
										
											2024-08-14 15:54:09 -03:00
										 |  |  |           )&.strip | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-22 18:32:42 -03:00
										 |  |  |         if result.present? && result.downcase.include?(search_for_text.downcase) | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |           user = User.find_by_username(canned_reply_user) if canned_reply_user.present? | 
					
						
							| 
									
										
										
										
											2025-03-06 17:18:15 +11:00
										 |  |  |           original_user = user | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |           user = user || Discourse.system_user | 
					
						
							| 
									
										
										
										
											2025-03-06 17:18:15 +11:00
										 |  |  |           if reply_persona_id.present? && action != :edit | 
					
						
							|  |  |  |             begin | 
					
						
							|  |  |  |               DiscourseAi::AiBot::Playground.reply_to_post( | 
					
						
							|  |  |  |                 post: post, | 
					
						
							|  |  |  |                 persona_id: reply_persona_id, | 
					
						
							|  |  |  |                 whisper: whisper, | 
					
						
							|  |  |  |                 user: original_user, | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |             rescue StandardError => e | 
					
						
							|  |  |  |               Discourse.warn_exception( | 
					
						
							|  |  |  |                 e, | 
					
						
							|  |  |  |                 message: "Error responding to: #{post&.url} in LlmTriage.handle", | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |               raise e if Rails.env.test? | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           elsif canned_reply.present? && action != :edit | 
					
						
							|  |  |  |             post_type = whisper ? Post.types[:whisper] : Post.types[:regular] | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |             PostCreator.create!( | 
					
						
							|  |  |  |               user, | 
					
						
							|  |  |  |               topic_id: post.topic_id, | 
					
						
							|  |  |  |               raw: canned_reply, | 
					
						
							|  |  |  |               reply_to_post_number: post.post_number, | 
					
						
							|  |  |  |               skip_validations: true, | 
					
						
							| 
									
										
										
										
											2025-03-06 17:18:15 +11:00
										 |  |  |               post_type: post_type, | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |             ) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           changes = {} | 
					
						
							|  |  |  |           changes[:category_id] = category_id if category_id.present? | 
					
						
							| 
									
										
										
										
											2024-12-11 11:19:44 -03:00
										 |  |  |           if SiteSetting.tagging_enabled? && tags.present? | 
					
						
							|  |  |  |             changes[:tags] = post.topic.tags.map(&:name).concat(tags) | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |           if changes.present? | 
					
						
							|  |  |  |             first_post = post.topic.posts.where(post_number: 1).first | 
					
						
							|  |  |  |             changes[:bypass_bump] = true | 
					
						
							|  |  |  |             changes[:skip_validations] = true | 
					
						
							|  |  |  |             first_post.revise(Discourse.system_user, changes) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           post.topic.update!(visible: false) if hide_topic | 
					
						
							| 
									
										
										
										
											2024-02-28 22:33:28 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-14 15:54:09 -03:00
										 |  |  |           if flag_post | 
					
						
							|  |  |  |             score_reason = | 
					
						
							|  |  |  |               I18n | 
					
						
							|  |  |  |                 .t("discourse_automation.scriptables.llm_triage.flagged_post") | 
					
						
							|  |  |  |                 .sub("%%LLM_RESPONSE%%", result) | 
					
						
							|  |  |  |                 .sub("%%AUTOMATION_ID%%", automation&.id.to_s) | 
					
						
							|  |  |  |                 .sub("%%AUTOMATION_NAME%%", automation&.name.to_s) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 07:19:56 +11:00
										 |  |  |             if flag_type == :spam || flag_type == :spam_silence | 
					
						
							| 
									
										
										
										
											2025-02-11 13:29:27 +10:00
										 |  |  |               result = | 
					
						
							|  |  |  |                 PostActionCreator.new( | 
					
						
							|  |  |  |                   Discourse.system_user, | 
					
						
							|  |  |  |                   post, | 
					
						
							|  |  |  |                   PostActionType.types[:spam], | 
					
						
							|  |  |  |                   message: score_reason, | 
					
						
							|  |  |  |                   queue_for_review: true, | 
					
						
							|  |  |  |                 ).perform | 
					
						
							| 
									
										
										
										
											2024-11-26 07:19:56 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-11 13:29:27 +10:00
										 |  |  |               if flag_type == :spam_silence | 
					
						
							|  |  |  |                 if result.success? | 
					
						
							|  |  |  |                   SpamRule::AutoSilence.new(post.user, post).silence_user | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                   Rails.logger.warn( | 
					
						
							|  |  |  |                     "llm_triage: unable to flag post as spam, post action failed for #{post.id} with error: '#{result.errors.full_messages.join(",").truncate(3000)}'", | 
					
						
							|  |  |  |                   ) | 
					
						
							|  |  |  |                 end | 
					
						
							|  |  |  |               end | 
					
						
							| 
									
										
										
										
											2024-10-04 15:11:30 +09:00
										 |  |  |             else | 
					
						
							|  |  |  |               reviewable = | 
					
						
							|  |  |  |                 ReviewablePost.needs_review!(target: post, created_by: Discourse.system_user) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               reviewable.add_score( | 
					
						
							|  |  |  |                 Discourse.system_user, | 
					
						
							|  |  |  |                 ReviewableScore.types[:needs_approval], | 
					
						
							|  |  |  |                 reason: score_reason, | 
					
						
							|  |  |  |                 force_review: true, | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2024-08-14 15:54:09 -03:00
										 |  |  |           end | 
					
						
							| 
									
										
										
										
											2023-12-19 12:04:15 +11:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |