Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add extensible log handler #155

Merged
merged 33 commits into from
Jan 10, 2025
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
b37f6f5
Add extensible log handler, redirect logs to a file in the user's pro…
tcdent Dec 19, 2024
0de9e47
Fix custom log levels, expand & cleanup tests.
tcdent Dec 20, 2024
8b712cb
Thoughts on additional log levels.
tcdent Dec 20, 2024
575cf98
Merge branch 'main' into logging
bboynton97 Dec 22, 2024
fe9172d
merge telemetry and logging
bboynton97 Dec 22, 2024
d0d8177
No more sys.exit outside of main.py. Raise exceptions on internal err…
tcdent Dec 30, 2024
e69384a
repo org update
bboynton97 Dec 22, 2024
773e017
sticker pack
bboynton97 Dec 22, 2024
38223e2
authenticate CLI with agentstack account
bboynton97 Dec 22, 2024
3e8e4a1
Fixed a bug, if entered agent name was empty.
tkrevh Dec 17, 2024
c677b33
Added tests for new functions.
tkrevh Dec 17, 2024
dcd9814
Moved CLI_ENTRY out of individual test files to cli_test_utils as it …
tkrevh Dec 19, 2024
3ecedb7
Environment variables are written commented-out if no value is set.
tcdent Dec 23, 2024
e1d2733
Update tool configs to use null values for placeholder environment va…
tcdent Dec 23, 2024
25aaa3d
Don't override `false`` values as None when re-parsing env vars
tcdent Dec 23, 2024
a88dba9
Document for project structure and tasks leading to 0.3 release
tcdent Dec 11, 2024
2e27aa0
Update project structure docs with progress made and future plans
tcdent Dec 13, 2024
b716ec8
Update v0.3 roadmap.
tcdent Dec 20, 2024
a0983d3
telem with user token
bboynton97 Dec 26, 2024
eef36ad
update footer social links to point to agentstack socials
tnguyen21 Dec 29, 2024
eae05ba
Merge regression
tcdent Dec 31, 2024
c795fab
Merge branch 'main' into logging
tcdent Dec 31, 2024
39e75ba
Missing imports
tcdent Dec 31, 2024
bfe6b0f
Correct main interaction to support installed binary (`main.main` get…
tcdent Dec 31, 2024
295b573
Prevent loggers from other modules from affecting the agenrstack logg…
tcdent Dec 31, 2024
4e80428
Resolve #175
tcdent Dec 31, 2024
29a562d
Fix main entrypoint to have congruency between module usage and bin s…
tcdent Dec 31, 2024
6081e04
Comments cleanup
tcdent Dec 31, 2024
c93b58e
Merge branch 'main' into logging
tcdent Jan 10, 2025
7b4bc54
Only write to log files that already exist. This prevents us from pre…
tcdent Jan 10, 2025
b6790fd
Migrate print statements to use agentstack.log
tcdent Jan 10, 2025
75b9411
Typo.
tcdent Jan 10, 2025
718cd28
Error message newlines.
tcdent Jan 10, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion agentstack/cli/agentstack_data.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
from typing import Optional

from agentstack.utils import clean_input, get_version
from agentstack.logger import log
from agentstack import log


class ProjectMetadata:
Expand Down
2 changes: 1 addition & 1 deletion agentstack/cli/cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
ProjectStructure,
CookiecutterData,
)
from agentstack.logger import log
from agentstack import log
from agentstack import conf
from agentstack.conf import ConfigFile
from agentstack.utils import get_package_path
Expand Down
6 changes: 3 additions & 3 deletions agentstack/cli/run.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@
MAIN_MODULE_NAME = "main"


def _format_friendy_error_message(exception: Exception):
def _format_friendly_error_message(exception: Exception):
"""
Projects will throw various errors, especially on first runs, so we catch
them here and print a more helpful message.
Expand Down Expand Up @@ -84,7 +84,7 @@ def _import_project_module(path: Path):
assert spec.loader is not None # appease type checker

project_module = importlib.util.module_from_spec(spec)
sys.path.append(str((path / MAIN_FILENAME).parent))
sys.path.insert(0, str((path / MAIN_FILENAME).parent))
spec.loader.exec_module(project_module)
return project_module

Expand Down Expand Up @@ -124,6 +124,6 @@ def run_project(command: str = 'run', debug: bool = False, cli_args: Optional[st
if debug:
raise exception
print(term_color("\nAn error occurred while running your project:\n", 'red'))
print(_format_friendy_error_message(exception))
print(_format_friendly_error_message(exception))
print(term_color("\nRun `agentstack run --debug` for a full traceback.", 'blue'))
sys.exit(1)
12 changes: 12 additions & 0 deletions agentstack/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,9 @@
DEFAULT_FRAMEWORK = "crewai"
CONFIG_FILENAME = "agentstack.json"

DEBUG: bool = False

# The path to the project directory ie. working directory.
PATH: Path = Path()


Expand All @@ -18,6 +21,15 @@ def set_path(path: Union[str, Path, None]):
PATH = Path(path) if path else Path()


def set_debug(debug: bool):
"""
Set the debug flag in the project's configuration for the session; does not
get saved to the project's configuration file.
"""
global DEBUG
DEBUG = debug


def get_framework() -> Optional[str]:
"""The framework used in the project. Will be available after PATH has been set
and if we are inside a project directory.
Expand Down
25 changes: 10 additions & 15 deletions agentstack/generation/tool_generation.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,11 +6,11 @@
import ast

from agentstack import conf
from agentstack import log
from agentstack.conf import ConfigFile
from agentstack.exceptions import ValidationError
from agentstack import frameworks
from agentstack import packaging
from agentstack.utils import term_color
from agentstack.tools import ToolConfig
from agentstack.generation import asttools
from agentstack.generation.files import EnvFile
Expand Down Expand Up @@ -83,7 +83,7 @@ def add_tool(tool_name: str, agents: Optional[list[str]] = []):
tool = ToolConfig.from_tool_name(tool_name)

if tool_name in agentstack_config.tools:
print(term_color(f'Tool {tool_name} is already installed', 'blue'))
log.notify(f'Tool {tool_name} is already installed')
else: # handle install
tool_file_path = tool.get_impl_file_path(agentstack_config.framework)

Expand All @@ -97,7 +97,7 @@ def add_tool(tool_name: str, agents: Optional[list[str]] = []):
with ToolsInitFile(conf.PATH / TOOLS_INIT_FILENAME) as tools_init:
tools_init.add_import_for_tool(tool, agentstack_config.framework)
except ValidationError as e:
print(term_color(f"Error adding tool:\n{e}", 'red'))
log.error(f"Error adding tool:\n{e}")

if tool.env: # add environment variables which don't exist
with EnvFile() as env:
Expand All @@ -117,20 +117,19 @@ def add_tool(tool_name: str, agents: Optional[list[str]] = []):
if not agents: # If no agents are specified, add the tool to all agents
agents = frameworks.get_agent_names()
for agent_name in agents:
print(f'Adding tool {tool.name} to agent {agent_name}')
log.info(f'Adding tool {tool.name} to agent {agent_name}')
frameworks.add_tool(tool, agent_name)

print(term_color(f'🔨 Tool {tool.name} added to agentstack project successfully', 'green'))
log.success(f'🔨 Tool {tool.name} added to agentstack project successfully')
if tool.cta:
print(term_color(f'🪩 {tool.cta}', 'blue'))
log.notify(f'🪩 {tool.cta}')


def remove_tool(tool_name: str, agents: Optional[list[str]] = []):
agentstack_config = ConfigFile()

if tool_name not in agentstack_config.tools:
print(term_color(f'Tool {tool_name} is not installed', 'red'))
sys.exit(1)
raise ValidationError(f'Tool {tool_name} is not installed')

tool = ToolConfig.from_tool_name(tool_name)
if tool.packages:
Expand All @@ -140,13 +139,13 @@ def remove_tool(tool_name: str, agents: Optional[list[str]] = []):
try:
os.remove(conf.PATH / f'src/tools/{tool.module_name}.py')
except FileNotFoundError:
print(f'"src/tools/{tool.module_name}.py" not found')
log.warning(f'"src/tools/{tool.module_name}.py" not found')

try: # Edit the user's project tool init file to exclude the tool
with ToolsInitFile(conf.PATH / TOOLS_INIT_FILENAME) as tools_init:
tools_init.remove_import_for_tool(tool, agentstack_config.framework)
except ValidationError as e:
print(term_color(f"Error removing tool:\n{e}", 'red'))
log.error(f"Error removing tool:\n{e}")

# Edit the framework entrypoint file to exclude the tool in the agent definition
if not agents: # If no agents are specified, remove the tool from all agents
Expand All @@ -161,8 +160,4 @@ def remove_tool(tool_name: str, agents: Optional[list[str]] = []):
with agentstack_config as config:
config.tools.remove(tool.name)

print(
term_color(f'🔨 Tool {tool_name}', 'green'),
term_color('removed', 'red'),
term_color('from agentstack project successfully', 'green'),
)
log.success(f'🔨 Tool {tool_name} removed from agentstack project successfully')
185 changes: 185 additions & 0 deletions agentstack/log.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
"""
`agentstack.log`

DEBUG: Detailed technical information, typically of interest when diagnosing problems.
TOOL_USE: A message to indicate the use of a tool.
THINKING: Information about an internal monologue or reasoning.
INFO: Useful information about the state of the application.
NOTIFY: A notification or update.
SUCCESS: An indication of a successful operation.
RESPONSE: A response to a request.
WARNING: An indication that something unexpected happened, but not severe.
ERROR: An indication that something went wrong, and the application may not be able to continue.

TODO TOOL_USE and THINKING are below INFO; this is intentional for now.

TODO would be cool to intercept all messages from the framework and redirect
them through this logger. This would allow us to capture all messages and display
them in the console and filter based on priority.

TODO With agentstack serve, we can direct all messages to the API, too.
"""

from typing import IO, Optional, Callable
import os, sys
import io
import logging
from agentstack import conf
from agentstack.utils import term_color

__all__ = [
'set_stdout',
'set_stderr',
'debug',
'tool_use',
'thinking',
'info',
'notify',
'success',
'response',
'warning',
'error',
]

LOG_NAME: str = 'agentstack'
LOG_FILENAME: str = 'agentstack.log'

# define additional log levels to accommodate other messages inside the app
DEBUG = logging.DEBUG # 10
TOOL_USE = 16
THINKING = 18
INFO = logging.INFO # 20
NOTIFY = 22
SUCCESS = 24
RESPONSE = 26
WARNING = logging.WARNING # 30
ERROR = logging.ERROR # 40

logging.addLevelName(THINKING, 'THINKING')
logging.addLevelName(TOOL_USE, 'TOOL_USE')
logging.addLevelName(NOTIFY, 'NOTIFY')
logging.addLevelName(SUCCESS, 'SUCCESS')
logging.addLevelName(RESPONSE, 'RESPONSE')

# `instance` is lazy so we have time to set up handlers
instance: Optional[logging.Logger] = None

stdout: IO = io.StringIO()
stderr: IO = io.StringIO()


def set_stdout(stream: IO):
"""
Redirect standard output messages to the given stream.
In practice, if a shell is available, pass: `sys.stdout`.
But, this can be any stream that implements the `write` method.
"""
global stdout, instance
stdout = stream
instance = None # force re-initialization


def set_stderr(stream: IO):
"""
Redirect standard error messages to the given stream.
In practice, if a shell is available, pass: `sys.stderr`.
But, this can be any stream that implements the `write` method.
"""
global stderr, instance
stderr = stream
instance = None # force re-initialization


def _create_handler(levelno: int) -> Callable:
"""Get the logging handler for the given log level."""

def handler(msg, *args, **kwargs):
global instance
if instance is None:
instance = _build_logger()
return instance.log(levelno, msg, *args, **kwargs)

return handler


debug = _create_handler(DEBUG)
tool_use = _create_handler(TOOL_USE)
thinking = _create_handler(THINKING)
info = _create_handler(INFO)
notify = _create_handler(NOTIFY)
success = _create_handler(SUCCESS)
response = _create_handler(RESPONSE)
warning = _create_handler(WARNING)
error = _create_handler(ERROR)


class ConsoleFormatter(logging.Formatter):
"""Formats log messages for display in the console."""

default_format = logging.Formatter('%(message)s')
formats = {
DEBUG: logging.Formatter('DEBUG: %(message)s'),
SUCCESS: logging.Formatter(term_color('%(message)s', 'green')),
NOTIFY: logging.Formatter(term_color('%(message)s', 'blue')),
WARNING: logging.Formatter(term_color('%(message)s', 'yellow')),
ERROR: logging.Formatter(term_color('%(message)s', 'red')),
}

def format(self, record: logging.LogRecord) -> str:
template = self.formats.get(record.levelno, self.default_format)
return template.format(record)


class FileFormatter(logging.Formatter):
"""Formats log messages for display in a log file."""

default_format = logging.Formatter('%(levelname)s: %(message)s')
formats = {
DEBUG: logging.Formatter('DEBUG (%(asctime)s):\n %(pathname)s:%(lineno)d\n %(message)s'),
}

def format(self, record: logging.LogRecord) -> str:
template = self.formats.get(record.levelno, self.default_format)
return template.format(record)


def _build_logger() -> logging.Logger:
"""
Build the logger with the appropriate handlers.
All log messages are written to the log file.
Errors and above are written to stderr if a stream has been configured.
Warnings and below are written to stdout if a stream has been configured.
"""
# global stdout, stderr

log = logging.getLogger(LOG_NAME)
# min log level set here cascades to all handlers
log.setLevel(DEBUG if conf.DEBUG else INFO)

# `conf.PATH`` can change during startup, so defer building the path
log_filename = conf.PATH / LOG_FILENAME
if not os.path.exists(log_filename):
os.makedirs(log_filename.parent, exist_ok=True)
log_filename.touch()

file_handler = logging.FileHandler(log_filename)
file_handler.setFormatter(FileFormatter())
file_handler.setLevel(DEBUG)
log.addHandler(file_handler)

# stdout handler for warnings and below
# `stdout` can change, so defer building the stream until we need it
stdout_handler = logging.StreamHandler(stdout)
stdout_handler.setFormatter(ConsoleFormatter())
stdout_handler.setLevel(DEBUG)
stdout_handler.addFilter(lambda record: record.levelno < ERROR)
log.addHandler(stdout_handler)

# stderr handler for errors and above
# `stderr` can change, so defer building the stream until we need it
stderr_handler = logging.StreamHandler(stderr)
stderr_handler.setFormatter(ConsoleFormatter())
stderr_handler.setLevel(ERROR)
log.addHandler(stderr_handler)

return log
30 changes: 0 additions & 30 deletions agentstack/logger.py

This file was deleted.

Loading
Loading