|
1 | 1 | import time |
| 2 | +from types import SimpleNamespace |
2 | 3 | from typing import Any, cast |
3 | 4 | from unittest.mock import Mock |
4 | 5 |
|
5 | 6 | import pytest |
6 | 7 |
|
7 | | -from graphon.model_runtime.entities.llm_entities import LLMMode |
| 8 | +from graphon.model_runtime.entities.llm_entities import LLMMode, LLMUsage |
8 | 9 | from graphon.model_runtime.entities.message_entities import PromptMessageRole |
| 10 | +from graphon.nodes.llm import llm_utils |
9 | 11 | from graphon.nodes.llm.entities import ModelConfig |
10 | 12 | from graphon.nodes.parameter_extractor import parameter_extractor_node |
11 | 13 | from graphon.nodes.parameter_extractor.entities import ( |
@@ -47,7 +49,12 @@ def _build_parameter_extractor_node() -> tuple[ParameterExtractorNode, VariableP |
47 | 49 | start_at=time.perf_counter(), |
48 | 50 | ) |
49 | 51 | init_params = build_graph_init_params(graph_config={"nodes": [], "edges": []}) |
50 | | - model_instance = Mock() |
| 52 | + model_instance = Mock( |
| 53 | + provider="test", |
| 54 | + model_name="test-model", |
| 55 | + parameters={}, |
| 56 | + stop=(), |
| 57 | + ) |
51 | 58 | prompt_message_serializer = Mock() |
52 | 59 | node = cast( |
53 | 60 | ParameterExtractorNode, |
@@ -151,6 +158,65 @@ def test_function_calling_prompt_template_renders_system_message() -> None: |
151 | 158 | assert prompt_messages[1].text == "Extract the location from this request." |
152 | 159 |
|
153 | 160 |
|
| 161 | +def test_prepare_run_context_exposes_model_identity_in_inputs( |
| 162 | + monkeypatch: pytest.MonkeyPatch, |
| 163 | +) -> None: |
| 164 | + node, variable_pool = _build_parameter_extractor_node() |
| 165 | + variable_pool.add(("start", "query"), "weather in sf") |
| 166 | + |
| 167 | + monkeypatch.setattr( |
| 168 | + llm_utils, |
| 169 | + "resolve_completion_params_variables", |
| 170 | + lambda parameters, _: parameters, |
| 171 | + ) |
| 172 | + monkeypatch.setattr( |
| 173 | + node, |
| 174 | + "_fetch_llm_model_schema", |
| 175 | + lambda **_: SimpleNamespace(features=[]), |
| 176 | + ) |
| 177 | + monkeypatch.setattr(node, "_build_run_prompt", lambda **_: ([], [])) |
| 178 | + |
| 179 | + run_context = node._prepare_run_context() # noqa: SLF001 |
| 180 | + |
| 181 | + assert run_context.inputs["query"] == "weather in sf" |
| 182 | + assert run_context.inputs["model_provider"] == "test" |
| 183 | + assert run_context.inputs["model_name"] == "test-model" |
| 184 | + |
| 185 | + |
| 186 | +def test_parameter_extractor_run_emits_model_identity_in_inputs( |
| 187 | + monkeypatch: pytest.MonkeyPatch, |
| 188 | +) -> None: |
| 189 | + node, variable_pool = _build_parameter_extractor_node() |
| 190 | + variable_pool.add(("start", "query"), "weather in sf") |
| 191 | + |
| 192 | + monkeypatch.setattr( |
| 193 | + llm_utils, |
| 194 | + "resolve_completion_params_variables", |
| 195 | + lambda parameters, _: parameters, |
| 196 | + ) |
| 197 | + monkeypatch.setattr( |
| 198 | + node, |
| 199 | + "_fetch_llm_model_schema", |
| 200 | + lambda **_: SimpleNamespace(features=[]), |
| 201 | + ) |
| 202 | + monkeypatch.setattr(node, "_build_run_prompt", lambda **_: ([], [])) |
| 203 | + |
| 204 | + invoke_result = SimpleNamespace( |
| 205 | + usage=LLMUsage.empty_usage(), |
| 206 | + message=SimpleNamespace( |
| 207 | + get_text_content=lambda: "{}", |
| 208 | + tool_calls=[], |
| 209 | + ), |
| 210 | + ) |
| 211 | + monkeypatch.setattr(node.model_instance, "invoke_llm", lambda **_: invoke_result) |
| 212 | + |
| 213 | + result = node._run() # noqa: SLF001 |
| 214 | + |
| 215 | + assert result.inputs["query"] == "weather in sf" |
| 216 | + assert result.inputs["model_provider"] == "test" |
| 217 | + assert result.inputs["model_name"] == "test-model" |
| 218 | + |
| 219 | + |
154 | 220 | def test_parameter_extractor_accepts_dependency_bundle() -> None: |
155 | 221 | variable_pool = build_variable_pool(variables=[]) |
156 | 222 | runtime_state = GraphRuntimeState( |
|
0 commit comments