Skip to content

fix: clean Hermes CLI ATIF telemetry#101

Merged
rapids-bot[bot] merged 5 commits into
NVIDIA:mainfrom
bbednarski9:bbednarski/nmf93-hermes-atif-telemetry-patch
May 13, 2026
Merged

fix: clean Hermes CLI ATIF telemetry#101
rapids-bot[bot] merged 5 commits into
NVIDIA:mainfrom
bbednarski9:bbednarski/nmf93-hermes-atif-telemetry-patch

Conversation

@bbednarski9
Copy link
Copy Markdown
Contributor

@bbednarski9 bbednarski9 commented May 13, 2026

Overview

This PR cleans up ATIF trajectories produced by end-to-end Hermes sessions through the NeMo Flow CLI. It removes noisy hook-only system rows, preserves useful hook context when Hermes emits sparse payloads, adds ancestry for hook-derived ATIF system steps, and avoids trailing empty ATIF steps at session boundaries.

  • I confirm this contribution is my own work, or I have the right to submit it under this project's license.
  • I searched existing issues and open pull requests, and this does not duplicate existing work.

Details

  • Treat Hermes on_session_end as a per-turn snapshot signal only. It now emits TurnEnded without also creating a user-visible HookMark system step.
  • Extend sidecar hook normalization to read Hermes event names and subagent identifiers from nested extra payload fields.
  • Preserve orphan Hermes subagent_stop evidence as a readable subagent_end_without_start ATIF mark and emit a warning when no matching subagent_start was observed.
  • Add ATIF mark-step metadata so hook-derived system steps include hook_event_name, ancestry, and invocation information.
  • Skip empty or null ATIF mark payloads so teardown/snapshot artifacts do not produce trailing {} system steps.
  • Add regression coverage for Hermes adapter normalization, per-turn ATIF snapshots, orphan subagent stops, empty hook marks, and generic ATIF mark-step lineage.

Validation run:

cargo fmt --check
git diff --check
cargo test -p nemo-flow observability::atif
cargo test -p nemo-flow-cli hermes
cargo test -p nemo-flow
cargo test -p nemo-flow-cli

Where should the reviewer start?

Start with crates/core/src/observability/atif.rs. The central design decision is that ATIF mark steps should remain useful system evidence, but empty marks should be dropped and non-empty marks should carry hook identity plus ancestry.

Then review crates/cli/src/adapters/hermes.rs for the Hermes-specific on_session_end behavior, and crates/cli/tests/coverage/session_tests.rs for the end-to-end sidecar regression cases.

Related Issues: (use one of the action keywords Closes / Fixes / Resolves / Relates to)

  • Closes #

Summary by CodeRabbit

  • New Features

    • Track API request errors with HTTP status/error details and richer request/response summaries.
    • Per-event payload fidelity detection and preference for exact sanitized payloads when available.
    • Broader detection of event names and subagent identifiers from nested payload fields.
  • Bug Fixes

    • Improved diagnostics for orphan subagent end events.
    • Per-turn session-end now emits a single turn-ended marker while preserving session scope for snapshots.
    • ATIF: skip empty marks and include ancestry/invocation details.
  • Tests

    • Expanded Hermes and ATIF coverage and added shared test helpers.
  • Documentation

    • Clarified ATIF snapshot behavior for Hermes hook lifecycles.

Review Change Stack

Signed-off-by: Bryan Bednarski <bbednarski@nvidia.com>
…missing ancestry on hook steps

Signed-off-by: Bryan Bednarski <bbednarski@nvidia.com>
@bbednarski9 bbednarski9 requested a review from a team as a code owner May 13, 2026 19:35
@coderabbitai
Copy link
Copy Markdown

coderabbitai Bot commented May 13, 2026

Walkthrough

Adds support for Hermes api_request_error, refactors per-turn session-end to emit only TurnEnded, implements per-event payload fidelity detection and exact request/response extraction, extends nested extra field fallbacks, and enriches ATIF mark steps with ancestry/invocation metadata; includes tests and installer update.

Changes

Hermes Hook Processing and ATIF Generation

Layer / File(s) Summary
Hermes adapter hook event and lifecycle handling
crates/cli/src/adapters/hermes.rs, crates/cli/src/session.rs, crates/cli/src/installer.rs
Adapter now handles new apirequesterror hook emitting LlmEnded; onsessionend refactored to early-return with only TurnEnded (removing duplicate post-classification append); orphan subagent-end logs a stderr diagnostic; api_request_error added to installer allowlist.
Payload exactness detection and request/response reconstruction
crates/cli/src/adapters/hermes.rs
Metadata derives per-event provider_payload_exact and fidelity_source; hermes_llm_request and hermes_llm_response prefer exact sanitized payloads when available, otherwise fall back to expanded summary objects (response summary gains status/retry/error fields); new helpers hermes_payload_exact, hermes_exact_request, hermes_exact_response, and is_truncated_payload added.
Extended fallback chains for event and agent identity extraction
crates/cli/src/adapters/mod.rs
event_name and subagent_id now recognize additional nested payload.extra.* variants (hook_event_name/eventName/agent_id/subagent.id patterns).
Adapter tests for lifecycle and exact payload mapping
crates/cli/tests/coverage/adapters_tests.rs
Updated on_session_end test to expect a single TurnEnded; added test for extracting hook_event_name and subagent_id from extra; adjusted fidelity/assertion to check provider_payload_exact in metadata; added tests mapping exact API hook payloads to LlmStarted/LlmEnded, mapping api_request_error to LlmEnded, and lossy handling for null requests.
Session and ATIF integration tests
crates/cli/tests/coverage/session_tests.rs
Added helper infrastructure and integration tests covering ATIF plugin output, idempotent duplicate AgentEnded handling, Hermes API metrics export, Hermes turn-end snapshot shape (no anonymous system step), orphan subagent_stop ancestry export, and skipping unknown/empty hook marks.
ATIF mark event generation with ancestry and invocation metadata
crates/core/src/observability/atif.rs
handle_mark now accepts EventLookupMaps, skips empty/null mark payloads, constructs AtifStepExtra.ancestry and AtifInvocationInfo, and emits system step messages via mark_message which injects hook_event_name when absent; added helpers is_empty_mark_payload, mark_message, and mark_hook_event_name.
Test infrastructure and ATIF unit tests
crates/cli/tests/coverage/setup_tests.rs, crates/core/tests/unit/atif_tests.rs
Setup test extended to assert api_request_error in merged Hermes config; TestEventBuilder::metadata() added; unit tests verify mark steps include hook_event_name and ancestry, and verify empty mark payloads are skipped with total_steps == Some(0).

🎯 3 (Moderate) | ⏱️ ~25 minutes

Suggested labels: DO NOT MERGE

🚥 Pre-merge checks | ✅ 5
✅ Passed checks (5 passed)
Check name Status Explanation
Description check ✅ Passed Description is well-structured with all required template sections completed, including overview, details, reviewer guidance, and validation steps.
Docstring Coverage ✅ Passed Docstring coverage is 91.49% which is sufficient. The required threshold is 80.00%.
Linked Issues check ✅ Passed Check skipped because no linked issues were found for this pull request.
Out of Scope Changes check ✅ Passed Check skipped because no linked issues were found for this pull request.
Title check ✅ Passed Title follows Conventional Commits format with type 'fix' and concise imperative summary, under 72 characters.

✏️ Tip: You can configure your own custom pre-merge checks in the settings.

✨ Finishing Touches
🧪 Generate unit tests (beta)
  • Create PR with unit tests

Comment @coderabbitai help to get the list of available commands and usage tips.

@github-actions github-actions Bot added size:L PR is large Bug issue describes bug; PR fixes bug lang:rust PR changes/introduces Rust code labels May 13, 2026
Copy link
Copy Markdown

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Inline comments:
In `@crates/cli/src/adapters/hermes.rs`:
- Around line 224-236: The hermes_exact_request function currently returns
Some(request) even when the extracted request is JSON null, causing
provider_payload_exact to be set for empty payloads; update hermes_exact_request
(which calls hermes_value_at and is_truncated_payload) to treat a null request
as non-exact by checking request.is_null() and returning None in that case
(i.e., only return Some(request) when the request is present, non-null, and not
truncated).
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: ASSERTIVE

Plan: Enterprise

Run ID: db74c87b-982d-418b-a70a-1a281d2b0221

📥 Commits

Reviewing files that changed from the base of the PR and between 704428b and db50515.

📒 Files selected for processing (9)
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/src/adapters/mod.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/core/tests/unit/atif_tests.rs
📜 Review details
⏰ Context from checks skipped due to timeout of 90000ms. You can increase the timeout in your CodeRabbit configuration to a maximum of 15 minutes (900000ms). (1)
  • GitHub Check: Check / Run
🧰 Additional context used
📓 Path-based instructions (12)
**/*.rs

📄 CodeRabbit inference engine (.agents/skills/test-ffi-surface/SKILL.md)

**/*.rs: Run cargo fmt --all for FFI work as it is Rust work
Run just test-rust for FFI validation
Run cargo clippy --workspace --all-targets -- -D warnings to enforce warnings-as-errors linting

**/*.rs: Run cargo fmt --all for Rust code formatting
Run cargo clippy --workspace --all-targets -- -D warnings to enforce Rust linting with no warnings
Run just test-rust as the shared-runtime build/test wrapper for Rust changes

Use Rust snake_case naming convention for Rust code

**/*.rs: Any Rust change must run just test-rust
Any Rust change must run cargo fmt --all
Any Rust change must run cargo clippy --workspace --all-targets -- -D warnings

**/*.rs: Run cargo fmt --all for Rust code formatting when Node changes touch Rust files
Run cargo clippy --workspace --all-targets -- -D warnings to enforce strict linting when Rust files changed as part of Node work

**/*.rs: Always run just test-rust when any Rust code changes
Always run cargo fmt --all when any Rust code changes
Always run cargo clippy --workspace --all-targets -- -D warnings when any Rust code changes

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/adapters/mod.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/src/adapters/hermes.rs
**/*.{rs,go,js,ts,tsx}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Include SPDX license headers in all Rust, Go, JavaScript, and TypeScript source files using C-style comment syntax

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/adapters/mod.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/src/adapters/hermes.rs
**/*.{rs,py,go,js,ts,tsx}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Use SONAR_IGNORE_START / SONAR_IGNORE_END markers only for documented false positives that cannot be resolved in code; keep ignored blocks small, add explanatory comments, and require reviewer sign-off

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/adapters/mod.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/src/adapters/hermes.rs
**/*.{js,ts,tsx,jsx,py,rs,go,java,c,cpp,h,cc,cxx,cs,rb,php,swift,kt}

📄 CodeRabbit inference engine (.agents/skills/prepare-pr/SKILL.md)

Changed files must be formatted with the language-native formatter

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/adapters/mod.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/src/adapters/hermes.rs
**/*.{py,js,ts,tsx,go,rs,md}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

Format changed files with the language-native formatter before the final lint/test pass

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/adapters/mod.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/src/adapters/hermes.rs
**/*.{rs,py,js,ts,tsx,go}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

During iteration, prefer uv run pre-commit run --files <changed files...> for targeted validation

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/src/adapters/mod.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/src/adapters/hermes.rs
{crates/**/tests/**,python/tests/**,go/nemo_flow/**/*_test.go}

⚙️ CodeRabbit configuration file

{crates/**/tests/**,python/tests/**,go/nemo_flow/**/*_test.go}: Tests should cover the behavior promised by the changed API surface, including error paths and cross-request isolation where relevant.
Prefer assertions on lifecycle events, scope stacks, middleware ordering, and binding parity over shallow smoke tests.

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/core/tests/unit/atif_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/tests/coverage/adapters_tests.rs
crates/{core,adaptive}/**

📄 CodeRabbit inference engine (.agents/skills/prepare-pr/SKILL.md)

Changes to crates/core or crates/adaptive must run the full language matrix

Files:

  • crates/core/tests/unit/atif_tests.rs
  • crates/core/src/observability/atif.rs
{crates/core,crates/adaptive}/**/*.rs

📄 CodeRabbit inference engine (.agents/skills/test-wasm-binding/SKILL.md)

If the change touched shared runtime semantics in crates/core or crates/adaptive, also use validate-change

Files:

  • crates/core/tests/unit/atif_tests.rs
  • crates/core/src/observability/atif.rs
crates/core/**/*.rs

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

When crates/core changes, run the full validation matrix across Rust, Python, Go, Node.js, and WebAssembly

crates/core/**/*.rs: Use Json = serde_json::Value in Rust-facing runtime APIs where the existing code expects JSON payloads.
Use Result<T> with FlowError in core runtime paths. Keep errors explicit and binding-appropriate at the wrapper layer.

Files:

  • crates/core/tests/unit/atif_tests.rs
  • crates/core/src/observability/atif.rs
crates/{core,adaptive}/**/*.rs

⚙️ CodeRabbit configuration file

crates/{core,adaptive}/**/*.rs: Review the Rust runtime for async correctness, scope isolation, middleware ordering, and event lifecycle regressions.
Pay close attention to task-local/thread-local scope propagation, callback lifetimes, stream finalization, and root_uuid isolation.
Public API changes should preserve existing behavior unless tests and docs show the intended migration path.

Files:

  • crates/core/tests/unit/atif_tests.rs
  • crates/core/src/observability/atif.rs
crates/core/src/observability/{atif,otel,openinference}.rs

📄 CodeRabbit inference engine (.agents/skills/maintain-observability/SKILL.md)

crates/core/src/observability/{atif,otel,openinference}.rs: When changing event fields, exporter behavior, subscriber config, or binding parity for ATIF, OpenTelemetry, or OpenInference, ensure the core event model and emitted fields remain in sync across ATIF (crates/core/src/observability/atif.rs), OpenTelemetry (crates/core/src/observability/otel.rs), and OpenInference (crates/core/src/observability/openinference.rs)
When event fields in observability exporters change, run the affected Rust crate tests plus just test-rust

Files:

  • crates/core/src/observability/atif.rs
🔇 Additional comments (15)
crates/cli/src/installer.rs (1)

63-65: LGTM!

crates/cli/src/adapters/mod.rs (1)

72-77: LGTM!

Also applies to: 315-319

crates/core/src/observability/atif.rs (1)

943-977: LGTM!

Also applies to: 1053-1053, 1061-1086

crates/core/tests/unit/atif_tests.rs (1)

54-57: LGTM!

Also applies to: 626-701

crates/cli/src/adapters/hermes.rs (1)

41-64: LGTM!

Also applies to: 66-222, 238-289

crates/cli/tests/coverage/setup_tests.rs (1)

335-335: LGTM!

crates/cli/src/session.rs (1)

731-734: LGTM!

crates/cli/tests/coverage/session_tests.rs (3)

361-442: LGTM!


444-497: LGTM!


498-540: LGTM!

crates/cli/tests/coverage/adapters_tests.rs (5)

320-324: LGTM!


339-360: LGTM!


397-397: LGTM!


433-524: LGTM!


526-564: LGTM!

Comment thread crates/cli/src/adapters/hermes.rs
@willkill07 willkill07 added this to the 0.2 milestone May 13, 2026
Comment thread crates/cli/src/session.rs Outdated
Comment thread crates/core/src/observability/atif.rs
@AjayThorve
Copy link
Copy Markdown
Contributor

AjayThorve commented May 13, 2026

Before merge, we should also do the docs cleanup:

docs/integrate-frameworks/coding-agent-hermes.md still says on_session_end is treated as a “per-turn mark” and implies ATIF is written only on on_session_finalize / on_session_reset. That should be
updated to match the new behavior: on_session_end writes a per-turn ATIF snapshot but does not emit a visible trajectory mark.

Copy link
Copy Markdown

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

Caution

Some comments are outside the diff and can’t be posted inline due to platform limitations.

⚠️ Outside diff range comments (1)
crates/cli/tests/coverage/setup_tests.rs (1)

260-269: 🧹 Nitpick | 🔵 Trivial | ⚡ Quick win

Also assert api_request_error in home_yaml for ConfigScope::Both.

The new hook key is only checked in project_yaml; this test path writes both files.

Proposed test update
     let home_yaml = std::fs::read_to_string(home.path().join(".hermes/config.yaml")).unwrap();
     assert!(home_yaml.contains("nemo-flow hook-forward hermes"));
+    assert!(home_yaml.contains("api_request_error"));
As per coding guidelines, "Tests should cover the behavior promised by the changed API surface, including error paths and cross-request isolation where relevant."
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@crates/cli/tests/coverage/setup_tests.rs` around lines 260 - 269, The test
currently asserts that project_yaml contains "api_request_error" but does not
assert the same for home_yaml when ConfigScope::Both is exercised; update the
test in crates/cli/tests/coverage/setup_tests.rs to add an assertion after
reading home_yaml (the variable home_yaml) that it contains "api_request_error"
(similar to the existing assert on project_yaml) so both config files are
validated for the new hook key when ConfigScope::Both is used.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Inline comments:
In `@crates/cli/tests/coverage/session_tests.rs`:
- Around line 41-49: The test helper read_atif_for_session is matching ATIF
files by doing trajectory.to_string().contains(session_id), which can select the
wrong file; change the selection to be deterministic by matching the filename
(entry.file_name() or entry.path().file_name()) against the known pattern
format!("trajectory-{}.json", session_id) or, alternatively, parse each JSON
Value and select the one whose session id field exactly equals session_id (e.g.,
value["session_id"] == session_id) so the function reliably returns the intended
ATIF file.

---

Outside diff comments:
In `@crates/cli/tests/coverage/setup_tests.rs`:
- Around line 260-269: The test currently asserts that project_yaml contains
"api_request_error" but does not assert the same for home_yaml when
ConfigScope::Both is exercised; update the test in
crates/cli/tests/coverage/setup_tests.rs to add an assertion after reading
home_yaml (the variable home_yaml) that it contains "api_request_error" (similar
to the existing assert on project_yaml) so both config files are validated for
the new hook key when ConfigScope::Both is used.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: ASSERTIVE

Plan: Enterprise

Run ID: 852543c4-cead-45bf-8b14-d1aac8967aaa

📥 Commits

Reviewing files that changed from the base of the PR and between db50515 and ee1329f.

📒 Files selected for processing (4)
  • crates/cli/src/installer.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/tests/coverage/setup_tests.rs
📜 Review details
⏰ Context from checks skipped due to timeout of 90000ms. You can increase the timeout in your CodeRabbit configuration to a maximum of 15 minutes (900000ms). (1)
  • GitHub Check: Check / Run
🧰 Additional context used
📓 Path-based instructions (7)
**/*.rs

📄 CodeRabbit inference engine (.agents/skills/test-ffi-surface/SKILL.md)

**/*.rs: Run cargo fmt --all for FFI work as it is Rust work
Run just test-rust for FFI validation
Run cargo clippy --workspace --all-targets -- -D warnings to enforce warnings-as-errors linting

**/*.rs: Run cargo fmt --all for Rust code formatting
Run cargo clippy --workspace --all-targets -- -D warnings to enforce Rust linting with no warnings
Run just test-rust as the shared-runtime build/test wrapper for Rust changes

Use Rust snake_case naming convention for Rust code

**/*.rs: Any Rust change must run just test-rust
Any Rust change must run cargo fmt --all
Any Rust change must run cargo clippy --workspace --all-targets -- -D warnings

**/*.rs: Run cargo fmt --all for Rust code formatting when Node changes touch Rust files
Run cargo clippy --workspace --all-targets -- -D warnings to enforce strict linting when Rust files changed as part of Node work

**/*.rs: Always run just test-rust when any Rust code changes
Always run cargo fmt --all when any Rust code changes
Always run cargo clippy --workspace --all-targets -- -D warnings when any Rust code changes

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/tests/coverage/session_tests.rs
**/*.{rs,go,js,ts,tsx}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Include SPDX license headers in all Rust, Go, JavaScript, and TypeScript source files using C-style comment syntax

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/tests/coverage/session_tests.rs
**/*.{rs,py,go,js,ts,tsx}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Use SONAR_IGNORE_START / SONAR_IGNORE_END markers only for documented false positives that cannot be resolved in code; keep ignored blocks small, add explanatory comments, and require reviewer sign-off

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/tests/coverage/session_tests.rs
**/*.{js,ts,tsx,jsx,py,rs,go,java,c,cpp,h,cc,cxx,cs,rb,php,swift,kt}

📄 CodeRabbit inference engine (.agents/skills/prepare-pr/SKILL.md)

Changed files must be formatted with the language-native formatter

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/tests/coverage/session_tests.rs
**/*.{py,js,ts,tsx,go,rs,md}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

Format changed files with the language-native formatter before the final lint/test pass

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/tests/coverage/session_tests.rs
**/*.{rs,py,js,ts,tsx,go}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

During iteration, prefer uv run pre-commit run --files <changed files...> for targeted validation

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/src/installer.rs
  • crates/cli/tests/coverage/session_tests.rs
{crates/**/tests/**,python/tests/**,go/nemo_flow/**/*_test.go}

⚙️ CodeRabbit configuration file

{crates/**/tests/**,python/tests/**,go/nemo_flow/**/*_test.go}: Tests should cover the behavior promised by the changed API surface, including error paths and cross-request isolation where relevant.
Prefer assertions on lifecycle events, scope stacks, middleware ordering, and binding parity over shallow smoke tests.

Files:

  • crates/cli/tests/coverage/setup_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
🔇 Additional comments (3)
crates/cli/src/installer.rs (1)

63-65: LGTM!

crates/cli/src/session.rs (1)

657-660: LGTM!

crates/cli/tests/coverage/session_tests.rs (1)

7-39: LGTM!

Also applies to: 126-527

Comment thread crates/cli/tests/coverage/session_tests.rs
@bbednarski9 bbednarski9 changed the title fix: clean Hermes CLI ATIF telemetry fix: clean Hermes CLI ATIF telemetry and hook_event_name for previously missed mark events May 13, 2026
Signed-off-by: Bryan Bednarski <bbednarski@nvidia.com>
@coderabbitai coderabbitai Bot added the DO NOT MERGE PR should not be merged; see PR for details label May 13, 2026
@willkill07 willkill07 removed DO NOT MERGE PR should not be merged; see PR for details labels May 13, 2026
@bbednarski9 bbednarski9 requested a review from willkill07 May 13, 2026 21:34
Copy link
Copy Markdown

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Inline comments:
In `@crates/cli/src/session.rs`:
- Around line 657-660: Remove the direct eprintln! used to report out-of-order
subagent end events in the event-handling code (the call that prints
event.event_name and event.subagent_id) and rely on the existing observability
call self.mark() to record the condition; if you prefer to keep printable
diagnostics, instead replace the eprintln! with a short comment explaining it is
intentional transient CLI diagnostic output (or add tracing::warn! only if you
add the tracing dependency), but do not leave an unannotated eprintln! in the
handler.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: ASSERTIVE

Plan: Enterprise

Run ID: 0b9e8807-2773-4ea7-99b9-c1546b0b69eb

📥 Commits

Reviewing files that changed from the base of the PR and between ee1329f and 207d98b.

📒 Files selected for processing (6)
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/core/src/observability/atif.rs
  • docs/integrate-frameworks/coding-agent-hermes.md
📜 Review details
⏰ Context from checks skipped due to timeout of 90000ms. You can increase the timeout in your CodeRabbit configuration to a maximum of 15 minutes (900000ms). (16)
  • GitHub Check: WebAssembly / Test (macos-arm64)
  • GitHub Check: Go / Test (windows-arm64)
  • GitHub Check: WebAssembly / Test (windows-arm64)
  • GitHub Check: WebAssembly / Test (linux-arm64)
  • GitHub Check: WebAssembly / Test (windows-amd64)
  • GitHub Check: Rust / Test (windows-arm64)
  • GitHub Check: WebAssembly / Test (linux-amd64)
  • GitHub Check: Go / Test (windows-amd64)
  • GitHub Check: Rust / Test (windows-amd64)
  • GitHub Check: Rust / Test (macos-arm64)
  • GitHub Check: Python / Test (windows-amd64)
  • GitHub Check: Node.js / Test (windows-arm64)
  • GitHub Check: Node.js / Test (windows-amd64)
  • GitHub Check: Python / Test (macos-arm64)
  • GitHub Check: Python / Test (windows-arm64)
  • GitHub Check: Documentation / Build
🧰 Additional context used
📓 Path-based instructions (31)
**/*.rs

📄 CodeRabbit inference engine (.agents/skills/test-ffi-surface/SKILL.md)

**/*.rs: Run cargo fmt --all for FFI work as it is Rust work
Run just test-rust for FFI validation
Run cargo clippy --workspace --all-targets -- -D warnings to enforce warnings-as-errors linting

**/*.rs: Run cargo fmt --all for Rust code formatting
Run cargo clippy --workspace --all-targets -- -D warnings to enforce Rust linting with no warnings
Run just test-rust as the shared-runtime build/test wrapper for Rust changes

Use Rust snake_case naming convention for Rust code

**/*.rs: Any Rust change must run just test-rust
Any Rust change must run cargo fmt --all
Any Rust change must run cargo clippy --workspace --all-targets -- -D warnings

**/*.rs: Run cargo fmt --all for Rust code formatting when Node changes touch Rust files
Run cargo clippy --workspace --all-targets -- -D warnings to enforce strict linting when Rust files changed as part of Node work

**/*.rs: Always run just test-rust when any Rust code changes
Always run cargo fmt --all when any Rust code changes
Always run cargo clippy --workspace --all-targets -- -D warnings when any Rust code changes

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/core/src/observability/atif.rs
**/*.{rs,go,js,ts,tsx}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Include SPDX license headers in all Rust, Go, JavaScript, and TypeScript source files using C-style comment syntax

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/core/src/observability/atif.rs
**/*.{rs,py,go,js,ts,tsx}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Use SONAR_IGNORE_START / SONAR_IGNORE_END markers only for documented false positives that cannot be resolved in code; keep ignored blocks small, add explanatory comments, and require reviewer sign-off

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/core/src/observability/atif.rs
**/*.{js,ts,tsx,jsx,py,rs,go,java,c,cpp,h,cc,cxx,cs,rb,php,swift,kt}

📄 CodeRabbit inference engine (.agents/skills/prepare-pr/SKILL.md)

Changed files must be formatted with the language-native formatter

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/core/src/observability/atif.rs
**/*.{py,js,ts,tsx,go,rs,md}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

Format changed files with the language-native formatter before the final lint/test pass

Files:

  • crates/cli/src/session.rs
  • docs/integrate-frameworks/coding-agent-hermes.md
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/core/src/observability/atif.rs
**/*.{rs,py,js,ts,tsx,go}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

During iteration, prefer uv run pre-commit run --files <changed files...> for targeted validation

Files:

  • crates/cli/src/session.rs
  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/src/adapters/hermes.rs
  • crates/cli/tests/coverage/adapters_tests.rs
  • crates/core/src/observability/atif.rs
**/*.{md,rst,html,txt}

📄 CodeRabbit inference engine (.agents/skills/review-doc-style/assets/nvidia-style-brand-terminology.md)

**/*.{md,rst,html,txt}: Always spell NVIDIA in all caps. Do not use Nvidia, nvidia, nVidia, nVIDIA, or NV.
Use an NVIDIA before a noun because the name starts with an 'en' sound.
Do not add a registered trademark symbol after NVIDIA when referring to the company.
Use trademark symbols with product names only when the document type or legal guidance requires them.
Verify official capitalization, spacing, and hyphenation for product names.
Precede NVIDIA product names with NVIDIA on first mention when it is natural and accurate.
Do not rewrite product names for grammar or title-case rules.
Preserve third-party product names according to the owner's spelling.
Include the company name and full model qualifier on first use when it helps identify the model.
Preserve the official capitalization and punctuation of model names.
Use shorter family names only after the full name is established.
Spell out a term on first use and put the acronym in parentheses unless the acronym is widely understood by the intended audience.
Use the acronym on later mentions after it has been defined.
For long documents, reintroduce the full term if readers might lose context.
Form plurals of acronyms with s, not an apostrophe, such as GPUs.
In headings, common acronyms can remain abbreviated. Spell out the term in the first or second sentence of the body.
Common terms such as CPU, GPU, PC, API, and UI usually do not need to be spelled out for developer audiences.

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,rst,html}

📄 CodeRabbit inference engine (.agents/skills/review-doc-style/assets/nvidia-style-brand-terminology.md)

Link the first mention of a product name when the destination helps the reader.

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,rst,txt}

📄 CodeRabbit inference engine (.agents/skills/review-doc-style/assets/nvidia-style-guide.md)

**/*.{md,rst,txt}: Spell NVIDIA in all caps. Do not use Nvidia, nvidia, or NV.
Format commands, code elements, expressions, package names, file names, and paths as inline code.
Use descriptive link text. Avoid raw URLs and weak anchors such as 'here' or 'read more.'
Use title case consistently for technical documentation headings.
Introduce code blocks, lists, tables, and images with complete sentences.
Write procedures as imperative steps. Keep steps parallel and split long procedures into smaller tasks.
Prefer active voice, present tense, short sentences, contractions, and plain English.
Use can for possibility and reserve may for permission.
Use after for temporal relationships instead of once.
Prefer refer to over see when the wording points readers to another resource.
Avoid culture-specific idioms, unnecessary Latinisms, jokes, and marketing exaggeration in technical documentation.
Spell out months in body text, avoid ordinal dates, and use clear time zones.
Spell out whole numbers from zero through nine unless they are technical values, parameters, versions, or UI values.
Use numerals for 10 or greater and include commas in thousands.
Do not add trademark symbols to learning-oriented documentation unless the source, platform, or legal guidance explicitly requires them.
Do not add trademark symbols to NeMo Flow learning documentation by default.
Do not rewrite API names, package names, command flags, or code literals for style reasons.

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,markdown,rst}

📄 CodeRabbit inference engine (.agents/skills/review-doc-style/assets/nvidia-style-technical-docs.md)

**/*.{md,markdown,rst}: Use title case consistently in technical documentation headings
Avoid quotation marks, ampersands, and exclamation marks in headings
Keep product, event, research, and whitepaper names in their official title case
Use title case for table headers
Do not force social-media sentence case into technical docs
Use monospace formatting for code elements, commands, parameters, package names, and expressions
Use monospace formatting for directories, file names, and paths
Use angle brackets inside monospace for variables inside paths, such as /home/<username>/.login
Use quotation marks for error messages and strings in documentation
Use bold formatting for UI buttons, menus, fields, and labels in documentation
Use angle brackets between UI labels for menu paths, such as File > Save As
Use italics for new terms on first use in documentation
Use italics for publication titles in documentation
Use plain text formatting for keyboard shortcuts in documentation
Prefer [NVIDIA/NeMo](link) format for GitHub repository references over generic phrases like 'the GitHub repo'
Introduce every code block with a complete sentence
Do not make a code block complete the grammar of the previous sentence
Do not continue a sentence after a code block
Use syntax highlighting when the format supports it for code blocks
Avoid the word 'snippet' unless the surrounding docs already use it as a term of art
Keep inline method, function, and class references consistent with nearby docs, omitting empty parentheses for prose readability when no call is shown
Use descriptive anchor text that matches the destination title when possible for links
Avoid raw URLs in running text in documentation
Avoid generic link anchors such as 'here,' 'this page,' and 'read more' in documentation
Include the acronym in link text if a linked term includes an acronym
Do not link long sentences or multiple sentences in documentation
Avoid links that pull readers away from a procedure unles...

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{html,md}

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Include SPDX license headers in HTML and Markdown files using HTML comment syntax

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
docs/**/*.md

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Run ./scripts/build-docs.sh for documentation site changes

docs/**/*.md: Relevant getting-started or reference docs must be updated when examples change
Release-policy docs must point to GitHub Releases as the only release-history source of truth

docs/**/*.md: Use title case for headings in technical documentation
Introduce code blocks, tables, and lists with complete lead-in sentences in documentation

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.md

📄 CodeRabbit inference engine (CONTRIBUTING.md)

Run Markdown link checking via lychee for README.md, CONTRIBUTING.md, and docs/ through pre-commit hooks

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,markdown,py,sh,bash,js,ts,java,cpp,go,rust}

📄 CodeRabbit inference engine (.agents/skills/contribute-docs/SKILL.md)

Keep package names, repo references, and build commands current in documentation

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{RELEASING.md,CHANGELOG.md,docs/**/*.md}

📄 CodeRabbit inference engine (.agents/skills/contribute-docs/SKILL.md)

Keep release-process and release-notes guidance in repo-maintainer docs such as RELEASING.md, not as user-facing docs pages or CHANGELOG.md

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,markdown,py,sh,bash}

📄 CodeRabbit inference engine (.agents/skills/contribute-docs/SKILL.md)

Keep stable user-facing wrappers at scripts/ root in docs and examples; only point at namespaced helper paths when documenting internal maintenance work

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,markdown,py,sh,bash,js,ts,example}

📄 CodeRabbit inference engine (.agents/skills/contribute-docs/SKILL.md)

Example commands must match current package names and paths

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
docs/integrate-frameworks/**/*.md

📄 CodeRabbit inference engine (.agents/skills/contribute-integration/SKILL.md)

Documentation must be updated if activation or usage of the integration changed

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{scripts/*.sh,docs/**/*.md}

📄 CodeRabbit inference engine (.agents/skills/contribute-integration/SKILL.md)

Use root ./scripts/*.sh commands in docs and contributor guidance as documented, with implementations under scripts/third-party/

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{docs/**,examples/**,crates/adaptive/**,python/nemo_flow/**,go/nemo_flow/**,**/{example,component}.{ts,tsx,js,rs,py,go}}

📄 CodeRabbit inference engine (.agents/skills/maintain-optimizer/SKILL.md)

Any new adaptive component kind must have documentation, examples, and binding coverage across all supported languages

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{README*,CHANGELOG*,docs/**/*.{md,rst,txt},examples/**/*,*.md}

📄 CodeRabbit inference engine (.agents/skills/rename-surfaces/SKILL.md)

Update documentation, examples, and getting-started guides with new package/module/crate names after rename operations

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
**/*.{md,txt,rst}

📄 CodeRabbit inference engine (.agents/skills/review-doc-style/SKILL.md)

**/*.{md,txt,rst}: Ensure commands, package names, file paths, and APIs in documentation are correct and not stale; flag incorrect or outdated information as blocking issues
Ensure examples and procedures in documentation will execute successfully with current APIs and commands
Use consistent user-facing terminology throughout documentation that matches current repo terminology
Capitalize NVIDIA correctly in all documentation and public-facing text
Format code, commands, paths, and filenames as inline code (monospace) in documentation
Use descriptive anchor text for links instead of bare URLs or weak labels like 'here' in documentation
Prefer active voice, present tense, short sentences, and plain English in documentation
Structure documentation procedures as imperative steps that are easy to scan and not too long for a single sequence
Prefer 'after' instead of 'once' for temporal references in documentation
Use 'can' instead of 'may' when describing possibility (rather than permission) in documentation
Avoid ambiguous numeric dates and ordinal dates in documentation body text

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{README.md,docs/**/*.md,examples/**/*.{js,ts,py,go,rs}}

📄 CodeRabbit inference engine (.agents/skills/maintain-packaging/SKILL.md)

Keep documentation and examples synchronized with current install, import, and build commands

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{README.md,CONTRIBUTING.md,docs/**/*.md}

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

For docs-only changes, run targeted checks only if commands, package names, or examples changed. Use just docs for docs-site builds and just docs-linkcheck when links changed

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{docs/**,README.md,CONTRIBUTING.md,RELEASING.md,SECURITY.md}

⚙️ CodeRabbit configuration file

{docs/**,README.md,CONTRIBUTING.md,RELEASING.md,SECURITY.md}: Review documentation for technical accuracy against the current API, command correctness, and consistency across language bindings.
Flag stale examples, missing SPDX headers where required, and instructions that no longer match CI or pre-commit behavior.

Files:

  • docs/integrate-frameworks/coding-agent-hermes.md
{crates/**/tests/**,python/tests/**,go/nemo_flow/**/*_test.go}

⚙️ CodeRabbit configuration file

{crates/**/tests/**,python/tests/**,go/nemo_flow/**/*_test.go}: Tests should cover the behavior promised by the changed API surface, including error paths and cross-request isolation where relevant.
Prefer assertions on lifecycle events, scope stacks, middleware ordering, and binding parity over shallow smoke tests.

Files:

  • crates/cli/tests/coverage/session_tests.rs
  • crates/cli/tests/coverage/adapters_tests.rs
crates/core/src/observability/{atif,otel,openinference}.rs

📄 CodeRabbit inference engine (.agents/skills/maintain-observability/SKILL.md)

crates/core/src/observability/{atif,otel,openinference}.rs: When changing event fields, exporter behavior, subscriber config, or binding parity for ATIF, OpenTelemetry, or OpenInference, ensure the core event model and emitted fields remain in sync across ATIF (crates/core/src/observability/atif.rs), OpenTelemetry (crates/core/src/observability/otel.rs), and OpenInference (crates/core/src/observability/openinference.rs)
When event fields in observability exporters change, run the affected Rust crate tests plus just test-rust

Files:

  • crates/core/src/observability/atif.rs
crates/{core,adaptive}/**

📄 CodeRabbit inference engine (.agents/skills/prepare-pr/SKILL.md)

Changes to crates/core or crates/adaptive must run the full language matrix

Files:

  • crates/core/src/observability/atif.rs
{crates/core,crates/adaptive}/**/*.rs

📄 CodeRabbit inference engine (.agents/skills/test-wasm-binding/SKILL.md)

If the change touched shared runtime semantics in crates/core or crates/adaptive, also use validate-change

Files:

  • crates/core/src/observability/atif.rs
crates/core/**/*.rs

📄 CodeRabbit inference engine (.agents/skills/validate-change/SKILL.md)

When crates/core changes, run the full validation matrix across Rust, Python, Go, Node.js, and WebAssembly

crates/core/**/*.rs: Use Json = serde_json::Value in Rust-facing runtime APIs where the existing code expects JSON payloads.
Use Result<T> with FlowError in core runtime paths. Keep errors explicit and binding-appropriate at the wrapper layer.

Files:

  • crates/core/src/observability/atif.rs
crates/{core,adaptive}/**/*.rs

⚙️ CodeRabbit configuration file

crates/{core,adaptive}/**/*.rs: Review the Rust runtime for async correctness, scope isolation, middleware ordering, and event lifecycle regressions.
Pay close attention to task-local/thread-local scope propagation, callback lifetimes, stream finalization, and root_uuid isolation.
Public API changes should preserve existing behavior unless tests and docs show the intended migration path.

Files:

  • crates/core/src/observability/atif.rs
🔇 Additional comments (5)
crates/core/src/observability/atif.rs (1)

943-977: LGTM!

Also applies to: 1053-1053, 1061-1094

crates/cli/src/adapters/hermes.rs (1)

41-64: LGTM!

Also applies to: 93-106, 121-138, 170-205, 207-289

crates/cli/tests/coverage/session_tests.rs (1)

7-67: LGTM!

Also applies to: 143-543

crates/cli/tests/coverage/adapters_tests.rs (1)

320-324: LGTM!

Also applies to: 339-360, 397-397, 433-596

docs/integrate-frameworks/coding-agent-hermes.md (1)

123-123: LGTM!

Also applies to: 129-133

Comment thread crates/cli/src/session.rs
@willkill07 willkill07 changed the title fix: clean Hermes CLI ATIF telemetry and hook_event_name for previously missed mark events fix: clean Hermes CLI ATIF telemetry May 13, 2026
@willkill07
Copy link
Copy Markdown
Member

/merge

@rapids-bot rapids-bot Bot merged commit 10a689b into NVIDIA:main May 13, 2026
67 of 68 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

Bug issue describes bug; PR fixes bug lang:rust PR changes/introduces Rust code size:L PR is large

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants