From 258b4f9165622120c1551dd4c0957baeec2b8f2b Mon Sep 17 00:00:00 2001 From: Hugo Saporetti Junior Date: Wed, 23 Oct 2024 19:59:30 -0300 Subject: [PATCH] Response and agent fixes --- src/main/askai/__classpath__.py | 21 +++------ src/main/askai/__main__.py | 33 +++++++++----- .../core/commander/commands/history_cmd.py | 15 +++---- .../processors/splitter/splitter_executor.py | 5 ++- .../processors/splitter/splitter_pipeline.py | 43 ++++++++++++------- .../processors/splitter/splitter_result.py | 4 +- .../processors/splitter/splitter_states.py | 16 +++---- src/main/askai/core/router/task_agent.py | 4 +- .../askai/core/support/shared_instances.py | 2 + 9 files changed, 83 insertions(+), 60 deletions(-) diff --git a/src/main/askai/__classpath__.py b/src/main/askai/__classpath__.py index 0c858a46..345b8a4d 100644 --- a/src/main/askai/__classpath__.py +++ b/src/main/askai/__classpath__.py @@ -12,19 +12,18 @@ Copyright (c) 2024, HomeSetup """ -from hspylib.modules.application.exit_status import ExitStatus - -from askai.core.model.api_keys import ApiKeys -from clitt.core.term.commons import is_a_tty -from hspylib.core.metaclass.classpath import Classpath -from hspylib.core.tools.commons import is_debugging, parent_path, root_dir - import logging as log import os -import pydantic import sys import warnings +import pydantic +from hspylib.core.metaclass.classpath import Classpath +from hspylib.core.tools.commons import is_debugging, parent_path, root_dir +from hspylib.modules.application.exit_status import ExitStatus + +from askai.core.model.api_keys import ApiKeys + if not is_debugging(): warnings.simplefilter("ignore", category=FutureWarning) warnings.simplefilter("ignore", category=UserWarning) @@ -32,17 +31,11 @@ warnings.simplefilter("ignore", category=UserWarning) warnings.filterwarnings("ignore", module="chromadb.db.impl.sqlite") -if not is_a_tty(): - log.getLogger().setLevel(log.ERROR) -else: - log.getLogger().setLevel(log.INFO) - if not os.environ.get("USER_AGENT"): # The AskAI User Agent, required by the langchain framework ASKAI_USER_AGENT: str = "AskAI-User-Agent" os.environ["USER_AGENT"] = ASKAI_USER_AGENT - try: API_KEYS: ApiKeys = ApiKeys() except pydantic.v1.error_wrappers.ValidationError as err: diff --git a/src/main/askai/__main__.py b/src/main/askai/__main__.py index 118c8d0e..ae0602b8 100755 --- a/src/main/askai/__main__.py +++ b/src/main/askai/__main__.py @@ -12,9 +12,15 @@ Copyright (c) 2024, HomeSetup """ -from askai.__classpath__ import classpath -from askai.core.askai_configs import configs -from askai.core.enums.run_modes import RunModes + +import logging as log +import os +import re +import sys +from textwrap import dedent +from typing import Any, AnyStr, Optional + +import click from clitt.core.tui.tui_application import TUIApplication from hspylib.core.enums.charset import Charset from hspylib.core.tools.commons import syserr, to_bool @@ -23,14 +29,11 @@ from hspylib.modules.application.argparse.parser_action import ParserAction from hspylib.modules.application.exit_status import ExitStatus from hspylib.modules.application.version import Version -from textwrap import dedent -from typing import Any, AnyStr, Optional -import click -import logging as log -import os -import re -import sys +from askai.__classpath__ import classpath +from askai.core.askai_configs import configs +from askai.core.enums.run_modes import RunModes +from askai.core.support.shared_instances import LOGGER_NAME class Main(TUIApplication): @@ -47,9 +50,19 @@ class Main(TUIApplication): INSTANCE: "Main" + @staticmethod + def setup_logs() -> None: + """TODO""" + # FIXME: Move this code to hspylib Application FW + log.basicConfig(level=log.WARNING) + logger = log.getLogger(LOGGER_NAME) + logger.setLevel(log.INFO) + logger.propagate = False + def __init__(self, app_name: str): super().__init__(app_name, self.VERSION, self.DESCRIPTION.format(self.VERSION), resource_dir=self.RESOURCE_DIR) self._askai: Any + Main.setup_logs() @property def askai(self) -> Any: diff --git a/src/main/askai/core/commander/commands/history_cmd.py b/src/main/askai/core/commander/commands/history_cmd.py index f0bead82..d8f7e217 100644 --- a/src/main/askai/core/commander/commands/history_cmd.py +++ b/src/main/askai/core/commander/commands/history_cmd.py @@ -32,6 +32,7 @@ def context_list() -> None: """List the entries in the chat context window.""" if (all_context := shared.context) and (length := len(all_context)) > 0: + ln: str = os.linesep display_text(f"### Listing ALL ({length}) Chat Contexts:\n\n---\n\n") for c in all_context: ctx, ctx_val = c[0], c[1] @@ -39,15 +40,13 @@ def context_list() -> None: f"- {ctx} ({len(ctx_val)}/{all_context.max_context_size} " f"tk [{all_context.length(ctx)}/{all_context.token_limit}]) \n" + indent( - "\n".join( - [ - f'{i}. **{e.role.title()}:**\n\n{indent(e.content, " " * 4)}' + os.linesep - for i, e in enumerate(ctx_val, start=1) - ] + ln.join([ + f'{i}. **{e.role.title()}:**\n\n{indent(text_formatter.strip_format(e.content), " " * 4)}' + + os.linesep + for i, e in enumerate(ctx_val, start=1)] ), - " " * 4, - ) - ) + " " * 4 + ), markdown=False) display_text(f"> Hint: Type: '/context forget [context] to forget a it.") else: text_formatter.commander_print(f"%YELLOW% Context is empty %NC%") diff --git a/src/main/askai/core/processors/splitter/splitter_executor.py b/src/main/askai/core/processors/splitter/splitter_executor.py index 2eccdde5..c9beb966 100644 --- a/src/main/askai/core/processors/splitter/splitter_executor.py +++ b/src/main/askai/core/processors/splitter/splitter_executor.py @@ -42,8 +42,8 @@ def display(self, message: str) -> None: if configs.is_debug: self._console.print(message) - def run(self): - with self._console.status(msg.wait(), spinner="dots") as spinner: + def run(self) -> None: + with self._console.status(f"[green]{str(self.pipeline.state)}[/green]", spinner="dots") as spinner: while not self.pipeline.state == States.COMPLETE: self.pipeline.track_previous() if 1 < configs.max_router_retries < 1 + self.pipeline.failures[self.pipeline.state.value]: @@ -89,6 +89,7 @@ def run(self): case _: self.display(f"[red] Error: Machine halted before complete!({self.pipeline.state})[/red]") break + execution_status: bool = self.pipeline.previous != self.pipeline.state execution_status_str: str = ( f"{'[green]√[/green]' if execution_status else '[red]X[/red]'}" diff --git a/src/main/askai/core/processors/splitter/splitter_pipeline.py b/src/main/askai/core/processors/splitter/splitter_pipeline.py index fd0df45c..a92b74e9 100644 --- a/src/main/askai/core/processors/splitter/splitter_pipeline.py +++ b/src/main/askai/core/processors/splitter/splitter_pipeline.py @@ -14,9 +14,10 @@ """ import logging as log from collections import defaultdict -from typing import AnyStr +from typing import AnyStr, Optional from hspylib.core.preconditions import check_state +from hspylib.core.tools.validator import Validator from langchain_core.prompts import PromptTemplate from transitions import Machine @@ -31,7 +32,7 @@ from askai.core.processors.splitter.splitter_states import States from askai.core.processors.splitter.splitter_transitions import Transition, TRANSITIONS from askai.core.router.evaluation import eval_response, EVALUATION_GUIDE -from askai.core.support.shared_instances import shared +from askai.core.support.shared_instances import shared, LOGGER_NAME class SplitterPipeline: @@ -44,7 +45,7 @@ class SplitterPipeline: def __init__(self, question: AnyStr): self._transitions: list[Transition] = [t for t in TRANSITIONS] self._machine: Machine = Machine( - name="Taius-Coder", model=self, + name=LOGGER_NAME, model=self, initial=States.STARTUP, states=States, transitions=self._transitions, auto_transitions=False ) @@ -82,28 +83,28 @@ def question(self) -> str: return self.result.question @property - def last_query(self) -> str: - return self.responses[-1].query + def last_query(self) -> Optional[str]: + return self.responses[-1].query if self.responses else None @last_query.setter def last_query(self, value: str) -> None: - self.responses[-1].query = value + self.responses[-1].query = value if self.responses else None @property - def last_answer(self) -> str: - return self.responses[-1].answer + def last_answer(self) -> Optional[str]: + return self.responses[-1].answer if self.responses else None @last_answer.setter def last_answer(self, value: str) -> None: - self.responses[-1].answer = value + self.responses[-1].answer = value if self.responses else None @property - def last_accuracy(self) -> AccResponse: - return self.responses[-1].accuracy + def last_accuracy(self) -> Optional[AccResponse]: + return self.responses[-1].accuracy if self.responses else None @last_accuracy.setter def last_accuracy(self, value: AccResponse) -> None: - self.responses[-1].accuracy = value + self.responses[-1].accuracy = value if self.responses else None @property def plan(self) -> ActionPlan: @@ -173,7 +174,7 @@ def st_execute_task(self) -> bool: def st_accuracy_check(self) -> AccColor: """TODO""" - if self.last_query is None or self.last_answer is None: + if not Validator.has_no_nulls(self.last_query, self.last_answer): return AccColor.BAD # FIXME Hardcoded for now @@ -204,7 +205,19 @@ def st_accuracy_check(self) -> AccColor: return acc.acc_color def st_refine_answer(self) -> bool: - return actions.refine_answer(self.question, self.final_answer, self.last_accuracy) + """TODO""" + if refined := actions.refine_answer(self.question, self.final_answer, self.last_accuracy): + final_response: PipelineResponse = PipelineResponse(self.question, refined, self.last_accuracy) + self.responses.clear() + self.responses.append(final_response) + return True + return False def st_final_answer(self) -> bool: - return actions.wrap_answer(self.question, self.final_answer, self.model) + """TODO""" + if wrapped := actions.wrap_answer(self.question, self.final_answer, self.model): + final_response: PipelineResponse = PipelineResponse(self.question, wrapped, self.last_accuracy) + self.responses.clear() + self.responses.append(final_response) + return True + return False diff --git a/src/main/askai/core/processors/splitter/splitter_result.py b/src/main/askai/core/processors/splitter/splitter_result.py index 13b79ec3..e363c31e 100644 --- a/src/main/askai/core/processors/splitter/splitter_result.py +++ b/src/main/askai/core/processors/splitter/splitter_result.py @@ -23,9 +23,9 @@ class SplitterResult: plan: ActionPlan | None = None model: ModelResult | None = None - def final_response(self) -> str: + def final_response(self, acc_threshold: AccColor = AccColor.MODERATE) -> str: """TODO""" return os.linesep.join( list(map(lambda r: r.answer, filter( - lambda acc: acc.accuracy and acc.accuracy.acc_color.passed(AccColor.MODERATE), self.responses))) + lambda acc: acc.accuracy and acc.accuracy.acc_color.passed(acc_threshold), self.responses))) ) diff --git a/src/main/askai/core/processors/splitter/splitter_states.py b/src/main/askai/core/processors/splitter/splitter_states.py index 57c2551b..31832c0d 100644 --- a/src/main/askai/core/processors/splitter/splitter_states.py +++ b/src/main/askai/core/processors/splitter/splitter_states.py @@ -18,21 +18,21 @@ class States(Enumeration): """Enumeration of possible task splitter states.""" # fmt: off - NOT_STARTED = 'Not started' + NOT_STARTED = 'Thinking' - STARTUP = 'Processing Query' + STARTUP = 'Processing query' - MODEL_SELECT = 'Selecting Model' + MODEL_SELECT = 'Selecting model' - TASK_SPLIT = 'Splitting Tasks' + TASK_SPLIT = 'Creating execution plan' - ACC_CHECK = 'Checking Accuracy' + ACC_CHECK = 'Checking accuracy' - EXECUTE_TASK = 'Executing Task' + EXECUTE_TASK = 'Executing task' - REFINE_ANSWER = 'Refining Answer' + REFINE_ANSWER = 'Refining answer' - WRAP_ANSWER = 'Wrapping Answer' + WRAP_ANSWER = 'Wrapping answer' COMPLETE = 'Completed' # fmt: on diff --git a/src/main/askai/core/router/task_agent.py b/src/main/askai/core/router/task_agent.py index a5ac7567..c26a49f5 100644 --- a/src/main/askai/core/router/task_agent.py +++ b/src/main/askai/core/router/task_agent.py @@ -87,7 +87,7 @@ def _create_lc_agent(self, temperature: Temperature = Temperature.COLDEST) -> Ru tools=tools, max_iterations=configs.max_agent_retries, memory=chat_memory, - handle_parsing_errors=True, + handle_parsing_errors="Generate a JSON blob that is fully parseable using the Python `json` module.", max_execution_time=configs.max_agent_execution_time_seconds, verbose=configs.is_debug, ) @@ -105,6 +105,8 @@ def _exec_task(self, task: AnyStr) -> Optional[Output]: return lc_agent.invoke({"input": task}) except openai.APIError as err: log.error(str(err)) + except ValueError as err: + log.error(str(err)) return None diff --git a/src/main/askai/core/support/shared_instances.py b/src/main/askai/core/support/shared_instances.py index 1edb03f5..b8959ba8 100644 --- a/src/main/askai/core/support/shared_instances.py +++ b/src/main/askai/core/support/shared_instances.py @@ -38,6 +38,8 @@ import os +LOGGER_NAME: str = 'Askai-Taius' + class SharedInstances(metaclass=Singleton): """Provides access to shared instances."""