Log Viewer
Overview
Inspect View provides a convenient way to visualize evaluation logs, including drilling into message histories, scoring decisions, and additional metadata written to the log. Here’s what the main view of an evaluation log looks like:
Below we’ll describe how to get the most out of using Inspect View.
Note that this section covers interactively exploring log files. You can also use the EvalLog
API to compute on log files (e.g. to compare across runs or to more systematically traverse results). See the section on Eval Logs to learn more about how to process log files with code.
View Basics
To run Inspect View, use the inspect view
command:
$ inspect view
By default, inspect view
will use the configured log directory of the environment it is run from (e.g. ./logs
). You can specify an alternate log directory using --log-dir
,for example:
$ inspect view --log-dir ./experiment-logs
By default it will run on port 7575 (and kill any existing inspect view
using that port). If you want to run two instances of inspect view
you can specify an alternate port:
$ inspect view --log-dir ./experiment-logs --port 6565
You only need to run inspect view
once at the beginning of a session (as it will automatically update to show new evaluations when they are run).
Log History
You can view and navigate between a history of all evals in the log directory using the menu at the top right:
Sample Details
Click a sample to drill into its messages, scoring, and metadata.
Messages
The messages tab displays the message history. In this example we see that the model make two tool calls before answering (the final assistant message is not fully displayed for brevity):
Looking carefully at the message history (especially for agents or multi-turn solvers) is critically important for understanding how well your evaluation is constructed.
Scoring
The scoring tab shows additional details including the full input and full model explanation for answers:
Metadata
The metadata tab shows additional data made available by solvers, tools, an scorers (in this case the web_search()
tool records which URLs it visited to retrieve additional context):
Scores and Answers
Reliable, high quality scoring is a critical component of every evaluation, and developing custom scorers that deliver this can be challenging. One major difficulty lies in the free form text nature of model output: we have a very specific target we are comparing against and we sometimes need to pick the answer out of a sea of text. Model graded output introduces another set of challenges entirely.
For comparison based scoring, scorers typically perform two core tasks:
- Extract the answer from the model’s output; and
- Compare the extracted answer to the target.
A scorer can fail to correctly score output at either of these steps. Failing to extract an answer entirely can occur (e.g. due to a regex that’s not quite flexible enough) and as can failing to correctly identify equivalent answers (e.g. thinking that “1,242” is different from “1242.00” or that “Yes.” is different than “yes”).
You can use the log viewer to catch and evaluate these sorts of issues. For example, here we can see that we were unable to extract answers for a couple of questions that were scored incorrect:
It’s possible that these answers are legitimately incorrect. However it’s also possible that the correct answer is in the model’s output but just in a format we didn’t quite expect. In each case you’ll need to drill into the sample to investigate.
Answers don’t just appear magically, scorers need to produce them during scoring. The scorers built in to Inspect all do this, but when you create a custom scorer, you should be sure to always include an answer
in the Score
objects you return if you can. For example:
return Score(
="C" if extracted == target.text else "I",
value=extracted,
answer=state.output.completion
explanation )
If we only return the value
of “C” or “I” we’d lose the context of exactly what was being compared when the score was assigned.
Note there is also an explanation
field: this is also important, as it allows you to view the entire context from which the answer was extracted from.
Filtering and Sorting
It’s often useful to filter log entries by score (for example, to investigate whether incorrect answers are due to scorer issues or are true negatives). Use the Scores picker to filter by specific scores:
By default, samples are ordered (with all samples for an epoch presented in sequence). However you can also order by score, or order by samples (so you see all of the results for a given sample across all epochs presented together). Use the Sort picker to control this:
Viewing by sample can be especially valuable for diagnosing the sources of inconsistency (and determining whether they are inherent or an artifact of the evaluation methodology). Above we can see that sample 1 is incorrect in epoch 1 because of issue the model had with forming a correct function call.
Python Logging
Beyond the standard information included an eval log file, you may want to do additional console logging to assist with developing and debugging. Inspect installs a log handler that displays logging output above eval progress as well as saves it into the evaluation log file.
If you use the recommend practice of the Python logging
library for obtaining a logger your logs will interoperate well with Inspect. For example, here we developing a web search tool and want to log each time a query occurs:
# setup logger for this source file
= logging.getLogger(__name__)
logger
# log each time we see a web query
f"web query: {query}") logger.info(
All of these log entries will be included in the sample transcript.
Log Levels
The log levels and their applicability are described below (in increasing order of severity):
Level | Description |
---|---|
debug |
Detailed information, typically of interest only when diagnosing problems. |
trace |
Show trace messages for runtime actions (e.g. model calls, subprocess exec, etc.). |
http |
HTTP diagnostics including requests and response statuses |
info |
Confirmation that things are working as expected. |
warning |
or indicative of some problem in the near future (e.g. ‘disk space low’). The software is still working as expected. |
error |
Due to a more serious problem, the software has not been able to perform some function |
critical |
A serious error, indicating that the program itself may be unable to continue running. |
Default Levels
By default, messages of log level warning
and higher are printed to the console, and messages of log level info
and higher are included in the sample transcript. This enables you to include many calls to logger.info()
in your code without having them show by default, while also making them available in the log viewer should you need them.
If you’d like to see ‘info’ messages in the console as well, use the --log-level info
option:
$ inspect eval biology_qa.py --log-level info
You can use the --log-level-transcript
option to control what level is written to the sample transcript:
$ inspect eval biology_qa.py --log-level-transcript http
Note that you can also set the log levels using the INSPECT_LOG_LEVEL
and INSPECT_LOG_LEVEL_TRANSCRIPT
environment variables (which are often included in a .env configuration file).
External File
In addition to seeing the Python logging activity at the end of an eval run in the log viewer, you can also arrange to have Python logger entries written to an external file. Set the INSPECT_PY_LOGGER_FILE
environment variable to do this:
export INSPECT_PY_LOGGER_FILE=/tmp/inspect.log
You can set this in the shell or within your global .env
file. By default, messages of level info
and higher will be written to the log file. If you set your main --log-level
lower than that (e.g. to http
) then the log file will follow. To set a distinct log level for the file, set the INSPECT_PY_LOGGER_FILE
environment variable. For example:
export INSPECT_PY_LOGGER_LEVEL=http
Use tail --follow
to track the contents of the log file in realtime. For example:
tail --follow /tmp/inspect.log
Task Information
The Info panel of the log viewer provides additional meta-information about evaluation tasks, including dataset, solver, and scorer details, git revision, and model token usage:
Publishing
You can use the command inspect view bundle
(or the bundle_log_dir()
function from Python) to create a self contained directory with the log viewer and a set of logs for display. This directory can then be deployed to any static web server (GitHub Pages, S3 buckets, or Netlify, for example) to provide a standalone version of the viewer. For example, to bundle the logs
directory to a directory named logs-www
:
$ inspect view bundle --log-dir logs --output-dir logs-www
Or to bundle the default log folder (read from INSPECT_LOG_DIR
):
$ inspect view bundle --output-dir logs-www
By default, an existing output dir will NOT be overwritten. Specify the --overwrite
option to remove and replace an existing output dir:
$ inspect view bundle --output-dir logs-www --overwrite
Bundling the viewer and logs will produce an output directory with the following structure:
logs-www
└── index.html
└── robots.txt
└── assets
└── ..
└── logs
└── ..
- 1
- The root viewer HTML
- 2
- Excludes this site from being indexed
- 3
- Supporting assets for the viewer
- 4
- The logs to be displayed
Deploy this folder to a static webserver to publish the log viewer.
Other Notes
You may provide a default output directory for bundling the viewer in your
.env
file by setting theINSPECT_VIEW_BUNDLE_OUTPUT_DIR
variable.You may specify an S3 url as the target for bundled views. See the Amazon S3 section for additional information on configuring S3.
You can use the
inspect_ai.log.bundle_log_dir
function in Python directly to bundle the viewer and logs into an output directory.The bundled viewer will show the first log file by default. You may link to the viewer to show a specific log file by including the
log_file
URL parameter, for example:https://logs.example.com?log_file=<log_file>
The bundled output directory includes a
robots.txt
file to prevent indexing by web crawlers. If you deploy this folder outside of the root of your website then you would need to update your rootrobots.txt
accordingly to exclude the folder from indexing (this is required because web crawlers only readrobots.txt
from the root of the website not subdirectories).