mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-08 23:06:03 +00:00
* refactor: pipeline routing rules - add routed_by_rule bypass and diagnostic logging - Add routing rules editor (RoutingRulesEditor component) - Add routed_by_rule bypass logic in response rules - Add diagnostic logging for pipeline routing - Database migration for bot pipeline routing rules - Extract RoutingRulesEditor component from BotForm - Revert log levels to debug * feat: add message_has_element routing rule type Support routing by message element type (Image, Voice, File, Forward, Face, At, AtAll, Quote) with eq/neq operators. * test: add unit tests for pipeline routing rules 20 tests covering _match_operator (eq/neq/contains/not_contains/ starts_with/regex/invalid) and resolve_pipeline_uuid (launcher_type/ launcher_id/message_content/message_has_element/first-match-wins/ skip-invalid/default-operator). * fix(web): add missing 'message_has_element' to routing rule type validation The Zod schema and TypeScript type for PipelineRoutingRule.type were missing the 'message_has_element' variant, causing silent form validation failure when saving routing rules with this type. * feat: add pipeline discard functionality and localization support * feat(web): improve drag-and-drop with DragOverlay, add discard monitoring and pipeline icons - Add DragOverlay for smooth cursor-following drag in routing rules editor - Remove transition to eliminate redundant swap animation on drop - Record discarded messages in monitoring system via _record_discarded_message - Display pipeline name (Workflow icon) and runner name (Play icon) on session monitor messages - Show discard badge on discarded messages in session monitor - Add i18n translations for discarded/userMessage/botMessage * fix: ensure discarded messages appear in session monitor and improve icons - Create/update monitoring session for discarded messages so they show in the bot session monitor (was only inserting message rows, not sessions) - Use human-readable 'Discarded' as pipeline_name instead of '__discard__' - Change runner icon from Play to Bot for better AI Agent semantics * fix: merge discarded messages into same session and remove session-level pipeline name - Use LauncherTypes enum for session_id in discarded messages to match the format used by monitoring_helper (fixes duplicate sessions) - Don't overwrite session pipeline info on discard — a session can have messages from multiple pipelines - Remove pipeline_name from session list and chat header since it's now shown per-message and a session is no longer single-pipeline * fix(web): only show save button on config tab in bot detail page * fix(web): scroll to bottom after messages render in session monitor --------- Co-authored-by: RockChinQ <rockchinq@gmail.com>
63 lines
2.1 KiB
Python
63 lines
2.1 KiB
Python
from __future__ import annotations
|
|
|
|
|
|
from . import rule
|
|
|
|
from .. import stage, entities
|
|
from ...utils import importutil
|
|
|
|
import langbot_plugin.api.entities.builtin.pipeline.query as pipeline_query
|
|
|
|
from . import rules
|
|
|
|
importutil.import_modules_in_pkg(rules)
|
|
|
|
|
|
@stage.stage_class('GroupRespondRuleCheckStage')
|
|
class GroupRespondRuleCheckStage(stage.PipelineStage):
|
|
"""群组响应规则检查器
|
|
|
|
仅检查群消息是否符合规则。
|
|
"""
|
|
|
|
rule_matchers: list[rule.GroupRespondRule]
|
|
"""检查器实例"""
|
|
|
|
async def initialize(self, pipeline_config: dict):
|
|
"""初始化检查器"""
|
|
|
|
self.rule_matchers = []
|
|
|
|
for rule_matcher in rule.preregisetered_rules:
|
|
rule_inst = rule_matcher(self.ap)
|
|
await rule_inst.initialize()
|
|
self.rule_matchers.append(rule_inst)
|
|
|
|
async def process(self, query: pipeline_query.Query, stage_inst_name: str) -> entities.StageProcessResult:
|
|
if query.launcher_type.value != 'group': # 只处理群消息
|
|
return entities.StageProcessResult(result_type=entities.ResultType.CONTINUE, new_query=query)
|
|
|
|
# 通过路由规则明确指定的流水线,跳过群响应规则检查
|
|
if query.variables and query.variables.get('_routed_by_rule', False):
|
|
return entities.StageProcessResult(result_type=entities.ResultType.CONTINUE, new_query=query)
|
|
|
|
rules = query.pipeline_config['trigger']['group-respond-rules']
|
|
|
|
use_rule = rules
|
|
|
|
# TODO revert it
|
|
# if str(query.launcher_id) in rules:
|
|
# use_rule = rules[str(query.launcher_id)]
|
|
|
|
for rule_matcher in self.rule_matchers: # 任意一个匹配就放行
|
|
res = await rule_matcher.match(str(query.message_chain), query.message_chain, use_rule, query)
|
|
if res.matching:
|
|
query.message_chain = res.replacement
|
|
|
|
return entities.StageProcessResult(
|
|
result_type=entities.ResultType.CONTINUE,
|
|
new_query=query,
|
|
)
|
|
|
|
return entities.StageProcessResult(result_type=entities.ResultType.INTERRUPT, new_query=query)
|