From 0c2ddc0e96128574adf0a329d16e5fb354298b5b Mon Sep 17 00:00:00 2001 From: Jeremiah Lowin <153965+jlowin@users.noreply.github.com> Date: Tue, 8 Oct 2024 21:29:23 -0400 Subject: [PATCH 1/3] Add flag for removing all system messages --- src/controlflow/events/message_compiler.py | 23 +++++++++++++--------- src/controlflow/llm/rules.py | 3 +++ 2 files changed, 17 insertions(+), 9 deletions(-) diff --git a/src/controlflow/events/message_compiler.py b/src/controlflow/events/message_compiler.py index 6ed4d54d..4dc94174 100644 --- a/src/controlflow/events/message_compiler.py +++ b/src/controlflow/events/message_compiler.py @@ -113,19 +113,22 @@ def convert_system_messages( messages: list[BaseMessage], rules: LLMRules ) -> list[BaseMessage]: """ - Converts system messages to human messages if the LLM doesnt support system messages. + Converts system messages to human messages if the LLM doesnt support system messages, either at all or in the first position. """ - if not messages or not rules.require_system_message_first: - return messages new_messages = [] - for message in messages: + for i, message in enumerate(messages): if isinstance(message, SystemMessage): - new_messages.append( - HumanMessage( - content=f"ORCHESTRATOR: {message.content}", name=message.name + # if system messages are not supported OR if they must be first and this is not the first message + # THEN convert the message to a human message + if not rules.allow_system_messages or ( + i > 0 and rules.require_system_message_first + ): + new_messages.append( + HumanMessage( + content=f"ORCHESTRATOR: {message.content}", name=message.name + ) ) - ) else: new_messages.append(message) return new_messages @@ -249,7 +252,9 @@ def compile_to_messages(self, agent: "Agent") -> list[BaseMessage]: messages = break_up_consecutive_ai_messages(messages, rules=context.llm_rules) messages = format_message_name(messages, rules=context.llm_rules) + messages = system_prompt + messages + # this should go last messages = convert_system_messages(messages, rules=context.llm_rules) - return system_prompt + messages + return messages diff --git a/src/controlflow/llm/rules.py b/src/controlflow/llm/rules.py index 21f7596a..fed84d9c 100644 --- a/src/controlflow/llm/rules.py +++ b/src/controlflow/llm/rules.py @@ -22,6 +22,9 @@ class LLMRules(ControlFlowModel): # require at least one non-system message require_at_least_one_message: bool = False + # system messages are supported as a role + allow_system_messages: bool = True + # system messages can only be provided as the very first message in a thread require_system_message_first: bool = False From 0dcbe6edb527d650a8f6976638c9fcb47d482fa7 Mon Sep 17 00:00:00 2001 From: Jeremiah Lowin <153965+jlowin@users.noreply.github.com> Date: Tue, 8 Oct 2024 21:40:25 -0400 Subject: [PATCH 2/3] Ensure system messages are always added --- src/controlflow/events/message_compiler.py | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/src/controlflow/events/message_compiler.py b/src/controlflow/events/message_compiler.py index 4dc94174..aff21195 100644 --- a/src/controlflow/events/message_compiler.py +++ b/src/controlflow/events/message_compiler.py @@ -113,14 +113,14 @@ def convert_system_messages( messages: list[BaseMessage], rules: LLMRules ) -> list[BaseMessage]: """ - Converts system messages to human messages if the LLM doesnt support system messages, either at all or in the first position. + Converts system messages to human messages if the LLM doesnt support system + messages, either at all or in the first position. """ - new_messages = [] for i, message in enumerate(messages): if isinstance(message, SystemMessage): - # if system messages are not supported OR if they must be first and this is not the first message - # THEN convert the message to a human message + # If system messages are not supported OR if they must be first and + # this is not the first message, THEN convert the message to a human message if not rules.allow_system_messages or ( i > 0 and rules.require_system_message_first ): @@ -129,6 +129,9 @@ def convert_system_messages( content=f"ORCHESTRATOR: {message.content}", name=message.name ) ) + else: + # If the system message is allowed, add it as-is + new_messages.append(message) else: new_messages.append(message) return new_messages From 1cc4986f96c87f02aaeeceabea93336bc3e7f342 Mon Sep 17 00:00:00 2001 From: Jeremiah Lowin <153965+jlowin@users.noreply.github.com> Date: Tue, 8 Oct 2024 21:44:52 -0400 Subject: [PATCH 3/3] Update test_run.py --- tests/test_run.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/test_run.py b/tests/test_run.py index c3a2fdd0..e8c92a74 100644 --- a/tests/test_run.py +++ b/tests/test_run.py @@ -93,7 +93,7 @@ def test_min_failed(self): task2 = Task("Task 2") task3 = Task("Task 3") - with instructions("fail tasks 1 and 3"): + with instructions("fail tasks 1 and 3. Don't work on task 2."): run_tasks( [task1, task2, task3], run_until=AnyFailed(min_failed=2), @@ -157,7 +157,7 @@ async def test_min_failed(self): task2 = Task("Task 2") task3 = Task("Task 3") - with instructions("fail tasks 1 and 3"): + with instructions("fail tasks 1 and 3. Don't work on task 2."): await run_tasks_async( [task1, task2, task3], run_until=AnyFailed(min_failed=2),