From f70aaf2e45badfc880aff127b4ae1e70781736f5 Mon Sep 17 00:00:00 2001 From: jjallaire Date: Thu, 19 Sep 2024 13:59:49 -0400 Subject: [PATCH 1/5] workaround grpc fork log spam (#460) --- src/inspect_ai/model/_providers/providers.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/src/inspect_ai/model/_providers/providers.py b/src/inspect_ai/model/_providers/providers.py index 4f191c755..be232faf4 100644 --- a/src/inspect_ai/model/_providers/providers.py +++ b/src/inspect_ai/model/_providers/providers.py @@ -1,3 +1,5 @@ +import os + from inspect_ai._util.error import pip_dependency_error from inspect_ai._util.version import verify_required_version @@ -69,6 +71,10 @@ def vertex() -> type[ModelAPI]: PACKAGE = "google-cloud-aiplatform" MIN_VERSION = "1.59.0" + # workaround log spam + # https://github.com/ray-project/ray/issues/24917 + os.environ["GRPC_ENABLE_FORK_SUPPORT"] = "0" + # verify we have the package try: import vertexai # type: ignore # noqa: F401 @@ -90,6 +96,10 @@ def google() -> type[ModelAPI]: PACKAGE = "google-generativeai" MIN_VERSION = "0.8.1" + # workaround log spam + # https://github.com/ray-project/ray/issues/24917 + os.environ["GRPC_ENABLE_FORK_SUPPORT"] = "0" + # verify we have the package try: import google.generativeai # type: ignore # noqa: F401 From 7a81a8028754d6bb6b45bc768b3f6516d13fc998 Mon Sep 17 00:00:00 2001 From: Charles Teague Date: Thu, 19 Sep 2024 14:02:27 -0400 Subject: [PATCH 2/5] Markdown render string in info events (#458) Co-authored-by: jjallaire --- src/inspect_ai/_view/www/dist/assets/index.js | 15 ++++++++++++++- .../src/samples/transcript/InfoEventView.mjs | 17 ++++++++++++++++- 2 files changed, 30 insertions(+), 2 deletions(-) diff --git a/src/inspect_ai/_view/www/dist/assets/index.js b/src/inspect_ai/_view/www/dist/assets/index.js index 9fa13d25c..4c2d4e352 100644 --- a/src/inspect_ai/_view/www/dist/assets/index.js +++ b/src/inspect_ai/_view/www/dist/assets/index.js @@ -16327,9 +16327,22 @@ const JSONPanel = ({ data, style }) => { `; }; const InfoEventView = ({ id, event, style }) => { + const panels = []; + if (typeof event.data === "string") { + panels.push( + m$1`<${MarkdownDiv} + markdown=${event.data} + style=${{ margin: "0.5em 0" }} + />` + ); + } else { + panels.push( + m$1`<${JSONPanel} data=${event.data} style=${{ margin: "0.5em 0" }} />` + ); + } return m$1` <${EventPanel} id=${id} title="Info" icon=${ApplicationIcons.info} style=${style}> - <${JSONPanel} data=${event.data} style=${{ margin: "0.5em 0" }}/> + ${panels} `; }; const ScoreEventView = ({ id, event, style }) => { diff --git a/src/inspect_ai/_view/www/src/samples/transcript/InfoEventView.mjs b/src/inspect_ai/_view/www/src/samples/transcript/InfoEventView.mjs index 67d347063..a74b70ddd 100644 --- a/src/inspect_ai/_view/www/src/samples/transcript/InfoEventView.mjs +++ b/src/inspect_ai/_view/www/src/samples/transcript/InfoEventView.mjs @@ -3,6 +3,7 @@ import { html } from "htm/preact"; import { ApplicationIcons } from "../../appearance/Icons.mjs"; import { EventPanel } from "./EventPanel.mjs"; import { JSONPanel } from "../../components/JsonPanel.mjs"; +import { MarkdownDiv } from "../../components/MarkdownDiv.mjs"; /** * Renders the InfoEventView component. @@ -14,8 +15,22 @@ import { JSONPanel } from "../../components/JsonPanel.mjs"; * @returns {import("preact").JSX.Element} The component. */ export const InfoEventView = ({ id, event, style }) => { + const panels = []; + if (typeof event.data === "string") { + panels.push( + html`<${MarkdownDiv} + markdown=${event.data} + style=${{ margin: "0.5em 0" }} + />`, + ); + } else { + panels.push( + html`<${JSONPanel} data=${event.data} style=${{ margin: "0.5em 0" }} />`, + ); + } + return html` <${EventPanel} id=${id} title="Info" icon=${ApplicationIcons.info} style=${style}> - <${JSONPanel} data=${event.data} style=${{ margin: "0.5em 0" }}/> + ${panels} `; }; From a849b37a3720e9b469d675639411890a31bf4a70 Mon Sep 17 00:00:00 2001 From: Charles Teague Date: Thu, 19 Sep 2024 15:00:41 -0400 Subject: [PATCH 3/5] Feature/sample error display (#461) * remove extraneou log messages * Display Error Type in SampleList and Detail Fixes #459 --- src/inspect_ai/_view/www/dist/assets/index.js | 26 +++++++++++----- src/inspect_ai/_view/www/src/App.mjs | 2 -- .../_view/www/src/samples/SampleDisplay.mjs | 5 +++- .../_view/www/src/samples/SampleError.mjs | 30 ++++++++++++++++--- .../_view/www/src/samples/SampleList.mjs | 2 +- 5 files changed, 49 insertions(+), 16 deletions(-) diff --git a/src/inspect_ai/_view/www/dist/assets/index.js b/src/inspect_ai/_view/www/dist/assets/index.js index 4c2d4e352..9260ca941 100644 --- a/src/inspect_ai/_view/www/dist/assets/index.js +++ b/src/inspect_ai/_view/www/dist/assets/index.js @@ -17367,7 +17367,7 @@ const resolveValue = (value, evalEvents) => { } return value; }; -const SampleError = ({ align, style }) => { +const SampleError = ({ message, align, style }) => { align = align || "center"; return m$1`
{ height: FontSize.small }} /> -
Error
+
${errorType(message)}
`; }; -const FlatSampleError = ({ style }) => { +const FlatSampleError = ({ message, style }) => { return m$1`
{ height: FontSize.base }} > - Error + ${errorType(message)}
`; }; +const errorType = (message) => { + if (!message) { + return "Error"; + } + if (message.includes("(")) { + return message.split("(")[0]; + } + return "Error"; +}; const printHtml = (html2, css) => { const printWindow = window.open("", "", "height=600,width=800"); printWindow.document.write("Print"); @@ -17740,7 +17749,10 @@ const SampleSummary = ({ id, sample, style, sampleDescriptor }) => { } columns.push({ label: "Score", - value: sample.error ? m$1`<${FlatSampleError} style=${{ marginTop: "0.4rem" }} />` : sampleDescriptor == null ? void 0 : sampleDescriptor.selectedScore(sample).render(), + value: sample.error ? m$1`<${FlatSampleError} + message=${sample.error.message} + style=${{ marginTop: "0.4rem" }} + />` : sampleDescriptor == null ? void 0 : sampleDescriptor.selectedScore(sample).render(), size: "minmax(2em, auto)", center: true }); @@ -18219,7 +18231,7 @@ const SampleRow = ({ display: "flex" }} > - ${sample.error ? m$1`<${SampleError} />` : sampleDescriptor == null ? void 0 : sampleDescriptor.selectedScore(sample).render()} + ${sample.error ? m$1`<${SampleError} message=${sample.error.message} />` : sampleDescriptor == null ? void 0 : sampleDescriptor.selectedScore(sample).render()} `; @@ -21512,7 +21524,6 @@ function App({ api: api2, pollForLogs = true }) { setHeadersLoading(false); }, [logs, setStatus, setLogHeaders, setHeadersLoading]); y(async () => { - console.log({ logs }); const targetLog = logs.files[selected]; if (targetLog && (!currentLog || currentLog.name !== targetLog.name)) { try { @@ -21686,7 +21697,6 @@ function App({ api: api2, pollForLogs = true }) { const index = result.files.findIndex((val) => { return log_file.endsWith(val.name); }); - console.log({ result, log_file, index }); if (index > -1) { setSelected(index); } diff --git a/src/inspect_ai/_view/www/src/App.mjs b/src/inspect_ai/_view/www/src/App.mjs index 12d4df4a8..1e5590547 100644 --- a/src/inspect_ai/_view/www/src/App.mjs +++ b/src/inspect_ai/_view/www/src/App.mjs @@ -94,7 +94,6 @@ export function App({ api, pollForLogs = true }) { // Load a specific log useEffect(async () => { - console.log({ logs }); const targetLog = logs.files[selected]; if (targetLog && (!currentLog || currentLog.name !== targetLog.name)) { try { @@ -308,7 +307,6 @@ export function App({ api, pollForLogs = true }) { const index = result.files.findIndex((val) => { return log_file.endsWith(val.name); }); - console.log({ result, log_file, index }); if (index > -1) { setSelected(index); } diff --git a/src/inspect_ai/_view/www/src/samples/SampleDisplay.mjs b/src/inspect_ai/_view/www/src/samples/SampleDisplay.mjs index 0bbbc65e1..d902a5e1e 100644 --- a/src/inspect_ai/_view/www/src/samples/SampleDisplay.mjs +++ b/src/inspect_ai/_view/www/src/samples/SampleDisplay.mjs @@ -368,7 +368,10 @@ const SampleSummary = ({ id, sample, style, sampleDescriptor }) => { columns.push({ label: "Score", value: sample.error - ? html`<${FlatSampleError} style=${{ marginTop: "0.4rem" }} />` + ? html`<${FlatSampleError} + message=${sample.error.message} + style=${{ marginTop: "0.4rem" }} + />` : sampleDescriptor?.selectedScore(sample).render(), size: "minmax(2em, auto)", center: true, diff --git a/src/inspect_ai/_view/www/src/samples/SampleError.mjs b/src/inspect_ai/_view/www/src/samples/SampleError.mjs index 453890819..b145345b5 100644 --- a/src/inspect_ai/_view/www/src/samples/SampleError.mjs +++ b/src/inspect_ai/_view/www/src/samples/SampleError.mjs @@ -8,12 +8,14 @@ import { ApplicationIcons } from "../appearance/Icons.mjs"; * Component to display a styled error message. * * @param {Object} props - The component properties. + * @param {string} [props.message] - The error message * @param {string} [props.align="center"] - The alignment for the error message. Defaults to "center". * @param {Object} [props.style] - Styles to add for this component * @returns {import("preact").JSX.Element} The error component. */ -export const SampleError = ({ align, style }) => { +export const SampleError = ({ message, align, style }) => { align = align || "center"; + return html`
{ height: FontSize.small, }} /> -
Error
+
${errorType(message)}
`; }; @@ -40,10 +42,11 @@ export const SampleError = ({ align, style }) => { * Component to display a styled error message. * * @param {Object} props - The component properties. + * @param {string} [props.message] - The message to display * @param {Object} [props.style] - Styles to add for this component * @returns {import("preact").JSX.Element} The error component. */ -export const FlatSampleError = ({ style }) => { +export const FlatSampleError = ({ message, style }) => { return html`
{ height: FontSize.base, }} > - Error + ${errorType(message)}
`; }; + +/** + * Extracts the error type from a given message. + * If the message contains parentheses, it returns the substring before the first parenthesis. + * Otherwise, it returns "Error". + * + * @param {string | undefined} message - The error message from which to extract the type. + * @returns {string} The extracted error type or "Error" if not found. + */ +const errorType = (message) => { + if (!message) { + return "Error"; + } + + if (message.includes("(")) { + return message.split("(")[0]; + } + return "Error"; +}; diff --git a/src/inspect_ai/_view/www/src/samples/SampleList.mjs b/src/inspect_ai/_view/www/src/samples/SampleList.mjs index a4942d163..6430c5fdc 100644 --- a/src/inspect_ai/_view/www/src/samples/SampleList.mjs +++ b/src/inspect_ai/_view/www/src/samples/SampleList.mjs @@ -306,7 +306,7 @@ const SampleRow = ({ }} > ${sample.error - ? html`<${SampleError} />` + ? html`<${SampleError} message=${sample.error.message} />` : sampleDescriptor?.selectedScore(sample).render()} From 91f978e16d9becd49f34ea1ab6daf31bbdd24342 Mon Sep 17 00:00:00 2001 From: "J.J. Allaire" Date: Thu, 19 Sep 2024 16:38:43 -0400 Subject: [PATCH 4/5] docs updates --- docs/agents-api.qmd | 8 -------- docs/log-viewer.qmd | 17 +++++++++++++++-- src/inspect_ai/_util/constants.py | 1 + 3 files changed, 16 insertions(+), 10 deletions(-) diff --git a/docs/agents-api.qmd b/docs/agents-api.qmd index 49de9c07e..502f59fbe 100644 --- a/docs/agents-api.qmd +++ b/docs/agents-api.qmd @@ -216,14 +216,6 @@ Note that we don't `await` the subtasks when building up our list of `searches`. ### Forking {#sec-forking} -::: {.callout-note appearance="simple"} -Note that the `fork()` feature described below is available only in the development version of Inspect. You can install the development version with: - -``` bash -pip install git+https://github.com/ukgovernmentbeis/inspect_ai -``` -::: - Inspect's `fork()` function provids a convenient wrapper around a very common use of subtasks: running a `TaskState` against a set of solvers in parallel to explore different trajectories. For example, let's say you have a solver named `explore()` that takes `temperature` as a parameter. You might want to try the solver out with multiple temperature values and then continue on with the best result: diff --git a/docs/log-viewer.qmd b/docs/log-viewer.qmd index d324d1eb0..09c2f5643 100644 --- a/docs/log-viewer.qmd +++ b/docs/log-viewer.qmd @@ -123,7 +123,21 @@ You can see all of these log entries in the **Logging** tab: ![](images/inspect-view-logging.png){.border .lightbox fig-alt="The Logging panel of the Inspect log viewer, displaying several info log messages from the web search tool indicating what queries were executed by the tool."} -It is important to note that the Inspect View will show all log entries level `info` or higher. However, printing every `info` message to the console during an eval might be too distracting, so the default log level for printing is `warning`. If you change it to `info` then you'll also see these log messages in the console: +### 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. | +| `http` | HTTP diagnostics including requests and response statuses | +| `sandbox` | Show commands sent to manage and execute code in sandboxes. | +| `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. | + +: Inspect View will show all log entries level `info` or higher. However, printing every `info` message to the console during an eval might be too distracting, so the default log level for the console is `warning`. If you change it to `info` then you'll also see these log messages in the console: ``` bash $ inspect eval biology_qa.py --log-level info @@ -181,7 +195,6 @@ By default, an existing output dir will NOT be overwritten. Specify the `--overw $ inspect view bundle --output-dir logs-www --overwrite ``` - Bundling the viewer and logs will produce an output directory with the following structure: ``` bash diff --git a/src/inspect_ai/_util/constants.py b/src/inspect_ai/_util/constants.py index 2c737d71e..88e9d7d39 100644 --- a/src/inspect_ai/_util/constants.py +++ b/src/inspect_ai/_util/constants.py @@ -22,6 +22,7 @@ "INFO", "WARNING", "ERROR", + "CRITICAL", ] DEFAULT_LOG_LEVEL = "warning" DEFAULT_LOG_BUFFER_LOCAL = 10 From 573ac95fa99f1319d9ebb45e28022d506e64e629 Mon Sep 17 00:00:00 2001 From: "J.J. Allaire" Date: Thu, 19 Sep 2024 16:58:23 -0400 Subject: [PATCH 5/5] always show epochs in task args --- src/inspect_ai/_display/rich.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/inspect_ai/_display/rich.py b/src/inspect_ai/_display/rich.py index 03f23a7cf..d788e3d35 100644 --- a/src/inspect_ai/_display/rich.py +++ b/src/inspect_ai/_display/rich.py @@ -512,7 +512,7 @@ def task_config(profile: TaskProfile, generate_config: bool = True) -> str: config = config | dict(profile.generate_config.model_dump(exclude_none=True)) config_print: list[str] = [] for name, value in config.items(): - if name not in ["limit", "epochs", "model"]: + if name not in ["limit", "model"]: config_print.append(f"{name}: {value}") values = ", ".join(config_print) if values: