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

LLM Custom Attributes Context Manager API #1214

Merged
merged 32 commits into from
Oct 10, 2024
Merged
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
c84d105
Add new custom attribute context manager.
umaannamalai Sep 18, 2024
5a7c183
Add new custom attribute context manager.
umaannamalai Sep 18, 2024
df11858
Add test cases.
umaannamalai Sep 20, 2024
37e5b3b
Merge branch 'llm-custom-attrs-api' of github.com:newrelic/newrelic-p…
umaannamalai Sep 20, 2024
977f1b7
Cleanup files.
umaannamalai Sep 20, 2024
4f82456
Merge branch 'main' into llm-custom-attrs-api
umaannamalai Sep 20, 2024
a5c6f1e
Merge branch 'main' into llm-custom-attrs-api
umaannamalai Sep 25, 2024
15605f2
Merge branch 'main' into llm-custom-attrs-api
mergify[bot] Sep 25, 2024
79b5cbe
Merge branch 'main' into llm-custom-attrs-api
mergify[bot] Sep 26, 2024
f51c621
Add new custom attribute context manager.
umaannamalai Sep 18, 2024
f6a8c5d
Add test cases.
umaannamalai Sep 20, 2024
9cd6d6b
Add new custom attribute context manager.
umaannamalai Sep 26, 2024
db2ac31
Cleanup files.
umaannamalai Sep 20, 2024
ba684ed
Add contextvars.
umaannamalai Sep 26, 2024
34ae93e
Cleanup.
umaannamalai Sep 30, 2024
fa903ec
Merge branch 'main' into llm-custom-attrs-api
umaannamalai Oct 1, 2024
7a01097
Merge branch 'main' into llm-custom-attrs-api
mergify[bot] Oct 1, 2024
d926afe
Merge branch 'llm-custom-attrs-api' of github.com:newrelic/newrelic-p…
umaannamalai Oct 2, 2024
7d690f6
Merge branch 'llm-custom-attrs-api' of github.com:newrelic/newrelic-p…
umaannamalai Oct 2, 2024
b38c89b
Update error handling.
umaannamalai Oct 2, 2024
a8beff8
Merge branch 'main' into llm-custom-attrs-api
mergify[bot] Oct 2, 2024
18278db
Undo unnecessary formatting changes
hmstepanek Oct 7, 2024
1694844
Revert to transaction based storage of attrs.
umaannamalai Oct 8, 2024
1c27c2b
Merge branch 'llm-custom-attrs-api' of github.com:newrelic/newrelic-p…
umaannamalai Oct 8, 2024
76e7640
[Mega-Linter] Apply linters fixes
umaannamalai Oct 8, 2024
6edffd1
Merge branch 'main' into llm-custom-attrs-api
mergify[bot] Oct 8, 2024
beeb67f
Delete context attrs instead of nullifying.
umaannamalai Oct 8, 2024
4f7c555
Merge branch 'llm-custom-attrs-api' of github.com:newrelic/newrelic-p…
umaannamalai Oct 8, 2024
46d5521
Update tests.
umaannamalai Oct 9, 2024
7160696
Remove newline.
umaannamalai Oct 9, 2024
ef40ca6
Assert attribute deletion was successful.
umaannamalai Oct 9, 2024
182cc71
Merge branch 'main' into llm-custom-attrs-api
mergify[bot] Oct 9, 2024
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: 2 additions & 0 deletions newrelic/agent.py
Original file line number Diff line number Diff line change
Expand Up @@ -158,6 +158,7 @@ def __asgi_application(*args, **kwargs):
)
from newrelic.api.ml_model import set_llm_token_count_callback as __set_llm_token_count_callback
from newrelic.api.ml_model import wrap_mlmodel as __wrap_mlmodel
from newrelic.api.llm_custom_attributes import WithLlmCustomAttributes as __WithLlmCustomAttributes
from newrelic.api.profile_trace import ProfileTraceWrapper as __ProfileTraceWrapper
from newrelic.api.profile_trace import profile_trace as __profile_trace
from newrelic.api.profile_trace import wrap_profile_trace as __wrap_profile_trace
Expand Down Expand Up @@ -251,6 +252,7 @@ def __asgi_application(*args, **kwargs):
record_custom_event = __wrap_api_call(__record_custom_event, "record_custom_event")
record_log_event = __wrap_api_call(__record_log_event, "record_log_event")
record_ml_event = __wrap_api_call(__record_ml_event, "record_ml_event")
WithLlmCustomAttributes = __wrap_api_call(__WithLlmCustomAttributes, "WithLlmCustomAttributes")
accept_distributed_trace_payload = __wrap_api_call(
__accept_distributed_trace_payload, "accept_distributed_trace_payload"
)
Expand Down
49 changes: 49 additions & 0 deletions newrelic/api/llm_custom_attributes.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
# Copyright 2010 New Relic, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import contextvars
import functools
import logging
from newrelic.api.transaction import current_transaction

_logger = logging.getLogger(__name__)
custom_attr_context_var = contextvars.ContextVar("custom_attr_context_var", default={})


class WithLlmCustomAttributes(object):
def __init__(self, custom_attr_dict):
transaction = current_transaction()
if not custom_attr_dict or not isinstance(custom_attr_dict, dict):
raise TypeError("custom_attr_dict must be a non-empty dictionary. Received type: %s" % type(custom_attr_dict))

# Add "llm." prefix to all keys in attribute dictionary
context_attrs = {k if k.startswith("llm.") else f"llm.{k}": v for k, v in custom_attr_dict.items()}

self.attr_dict = context_attrs
self.transaction = transaction

def __enter__(self):
if not self.transaction:
_logger.warning("WithLlmCustomAttributes must be called within the scope of a transaction.")
return self

token = custom_attr_context_var.set(self.attr_dict)
self.transaction._custom_attr_context_var = custom_attr_context_var
return token

def __exit__(self, exc, value, tb):
if self.transaction:
custom_attr_context_var.set(None)
self.transaction._custom_attr_context_var = custom_attr_context_var

6 changes: 6 additions & 0 deletions newrelic/hooks/external_botocore.py
Original file line number Diff line number Diff line change
Expand Up @@ -787,6 +787,11 @@ def handle_chat_completion_event(transaction, bedrock_attrs):
custom_attrs_dict = transaction._custom_params
llm_metadata_dict = {key: value for key, value in custom_attrs_dict.items() if key.startswith("llm.")}

llm_context_attrs = getattr(transaction, "_custom_attr_context_var", None)
if llm_context_attrs:
context_attrs = llm_context_attrs.get()
llm_metadata_dict.update(context_attrs)

span_id = bedrock_attrs.get("span_id", None)
trace_id = bedrock_attrs.get("trace_id", None)
request_id = bedrock_attrs.get("request_id", None)
Expand Down Expand Up @@ -818,6 +823,7 @@ def handle_chat_completion_event(transaction, bedrock_attrs):
"response.choices.finish_reason": bedrock_attrs.get("response.choices.finish_reason", None),
"error": bedrock_attrs.get("error", None),
}

chat_completion_summary_dict.update(llm_metadata_dict)
chat_completion_summary_dict = {k: v for k, v in chat_completion_summary_dict.items() if v is not None}

Expand Down
7 changes: 6 additions & 1 deletion newrelic/hooks/mlmodel_langchain.py
Original file line number Diff line number Diff line change
Expand Up @@ -698,7 +698,7 @@ def _get_run_manager_info(transaction, run_args, instance, completion_id):
# metadata and tags are keys in the config parameter.
metadata = {}
metadata.update((run_args.get("config") or {}).get("metadata") or {})
# Do not report intenral nr_completion_id in metadata.
# Do not report internal nr_completion_id in metadata.
umaannamalai marked this conversation as resolved.
Show resolved Hide resolved
metadata = {key: value for key, value in metadata.items() if key != "nr_completion_id"}
tags = []
tags.extend((run_args.get("config") or {}).get("tags") or [])
Expand All @@ -709,6 +709,11 @@ def _get_llm_metadata(transaction):
# Grab LLM-related custom attributes off of the transaction to store as metadata on LLM events
custom_attrs_dict = transaction._custom_params
llm_metadata_dict = {key: value for key, value in custom_attrs_dict.items() if key.startswith("llm.")}
llm_context_attrs = getattr(transaction, "_custom_attr_context_var", None)
if llm_context_attrs:
context_attrs = llm_context_attrs.get()
llm_metadata_dict.update(context_attrs)

return llm_metadata_dict


Expand Down
34 changes: 27 additions & 7 deletions newrelic/hooks/mlmodel_openai.py
Original file line number Diff line number Diff line change
Expand Up @@ -436,6 +436,7 @@ def _handle_completion_success(transaction, linking_metadata, completion_id, kwa
trace_id = linking_metadata.get("trace.id")
request_message_list = kwargs.get("messages") or []
stream = kwargs.get("stream", False)

# Only if streaming and streaming monitoring is enabled and the response is not empty
# do we not exit the function trace.
if not stream or not settings.ai_monitoring.streaming.enabled or not return_val:
Expand Down Expand Up @@ -475,12 +476,16 @@ def _handle_completion_success(transaction, linking_metadata, completion_id, kwa
# openai._legacy_response.LegacyAPIResponse
response = json.loads(response.http_response.text.strip())

_record_completion_success(transaction, linking_metadata, completion_id, kwargs, ft, response_headers, response)
_record_completion_success(
transaction, linking_metadata, completion_id, kwargs, ft, response_headers, response
)
except Exception:
_logger.warning(RECORD_EVENTS_FAILURE_LOG_MESSAGE % traceback.format_exception(*sys.exc_info()))


def _record_completion_success(transaction, linking_metadata, completion_id, kwargs, ft, response_headers, response):
def _record_completion_success(
transaction, linking_metadata, completion_id, kwargs, ft, response_headers, response
):
span_id = linking_metadata.get("span.id")
trace_id = linking_metadata.get("trace.id")
try:
Expand Down Expand Up @@ -642,6 +647,7 @@ def _record_completion_error(transaction, linking_metadata, completion_id, kwarg
}
llm_metadata = _get_llm_attributes(transaction)
error_chat_completion_dict.update(llm_metadata)

transaction.record_custom_event("LlmChatCompletionSummary", error_chat_completion_dict)

output_message_list = []
Expand Down Expand Up @@ -780,15 +786,20 @@ def _record_events_on_stop_iteration(self, transaction):
self._nr_ft.__exit__(None, None, None)
try:
openai_attrs = getattr(self, "_nr_openai_attrs", {})

# If there are no openai attrs exit early as there's no data to record.
if not openai_attrs:
return

completion_id = str(uuid.uuid4())
response_headers = openai_attrs.get("response_headers") or {}
_record_completion_success(
transaction, linking_metadata, completion_id, openai_attrs, self._nr_ft, response_headers, None
transaction,
linking_metadata,
completion_id,
openai_attrs,
self._nr_ft,
response_headers,
None
)
except Exception:
_logger.warning(RECORD_EVENTS_FAILURE_LOG_MESSAGE % traceback.format_exception(*sys.exc_info()))
Expand All @@ -813,7 +824,10 @@ def _handle_streaming_completion_error(self, transaction, exc):
return
linking_metadata = get_trace_linking_metadata()
completion_id = str(uuid.uuid4())
_record_completion_error(transaction, linking_metadata, completion_id, openai_attrs, self._nr_ft, exc)

_record_completion_error(
transaction, linking_metadata, completion_id, openai_attrs, self._nr_ft, exc
)


class AsyncGeneratorProxy(ObjectProxy):
Expand Down Expand Up @@ -927,8 +941,14 @@ def is_stream(wrapped, args, kwargs):
def _get_llm_attributes(transaction):
"""Returns llm.* custom attributes off of the transaction."""
custom_attrs_dict = transaction._custom_params
llm_metadata = {key: value for key, value in custom_attrs_dict.items() if key.startswith("llm.")}
return llm_metadata
llm_metadata_dict = {key: value for key, value in custom_attrs_dict.items() if key.startswith("llm.")}

llm_context_attrs = getattr(transaction, "_custom_attr_context_var", None)
if llm_context_attrs:
context_attrs = llm_context_attrs.get()
llm_metadata_dict.update(context_attrs)

return llm_metadata_dict


def instrument_openai_api_resources_embedding(module):
Expand Down
48 changes: 48 additions & 0 deletions tests/agent_features/test_llm_custom_attributes.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
# Copyright 2010 New Relic, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import pytest

from newrelic.api.background_task import background_task
from newrelic.api.transaction import current_transaction
from newrelic.api.llm_custom_attributes import WithLlmCustomAttributes


@background_task()
def test_llm_custom_attributes():
transaction = current_transaction()
with WithLlmCustomAttributes({"test": "attr", "test1": "attr1"}):
assert transaction._custom_attr_context_var.get() == {"llm.test": "attr", "llm.test1": "attr1"}

assert transaction._custom_attr_context_var.get() is None


@pytest.mark.parametrize("context_attrs", (None, "not-a-dict"))
@background_task()
def test_llm_custom_attributes_no_attrs(context_attrs):
transaction = current_transaction()

with pytest.raises(TypeError):
with WithLlmCustomAttributes(context_attrs):
assert transaction._custom_attr_context_var.get() is None


@background_task()
def test_llm_custom_attributes_prefixed_attrs():
transaction = current_transaction()
with WithLlmCustomAttributes({"llm.test": "attr", "llm.test1": "attr1"}):
umaannamalai marked this conversation as resolved.
Show resolved Hide resolved
# Validate API does not prefix attributes that already begin with "llm."
assert transaction._custom_attr_context_var.get() == {"llm.test": "attr", "llm.test1": "attr1"}

assert transaction._custom_attr_context_var.get() is None
42 changes: 23 additions & 19 deletions tests/external_botocore/test_bedrock_chat_completion.py
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
disabled_ai_monitoring_streaming_settings,
events_sans_content,
events_sans_llm_metadata,
events_with_context_attrs,
llm_token_count_callback,
set_trace_info,
)
Expand All @@ -59,6 +60,7 @@

from newrelic.api.background_task import background_task
from newrelic.api.transaction import add_custom_attribute
from newrelic.api.llm_custom_attributes import WithLlmCustomAttributes
from newrelic.common.object_names import callable_name
from newrelic.hooks.external_botocore import MODEL_EXTRACTORS

Expand Down Expand Up @@ -161,7 +163,7 @@ def expected_invalid_access_key_error_events(model_id):
def test_bedrock_chat_completion_in_txn_with_llm_metadata(
set_trace_info, exercise_model, expected_events, expected_metrics
):
@validate_custom_events(expected_events)
@validate_custom_events(events_with_context_attrs(expected_events))
# One summary event, one user message, and one response message from the assistant
@validate_custom_event_count(count=3)
@validate_transaction_metrics(
Expand All @@ -180,7 +182,8 @@ def _test():
add_custom_attribute("llm.conversation_id", "my-awesome-id")
add_custom_attribute("llm.foo", "bar")
add_custom_attribute("non_llm_attr", "python-agent")
exercise_model(prompt=_test_bedrock_chat_completion_prompt, temperature=0.7, max_tokens=100)
with WithLlmCustomAttributes({"context": "attr"}):
exercise_model(prompt=_test_bedrock_chat_completion_prompt, temperature=0.7, max_tokens=100)

_test()

Expand Down Expand Up @@ -320,7 +323,7 @@ def _test():
def test_bedrock_chat_completion_error_invalid_model(
bedrock_server, set_trace_info, response_streaming, expected_metrics
):
@validate_custom_events(chat_completion_invalid_model_error_events)
@validate_custom_events(events_with_context_attrs(chat_completion_invalid_model_error_events))
@validate_error_trace_attributes(
"botocore.errorfactory:ValidationException",
exact_attrs={
Expand Down Expand Up @@ -350,22 +353,23 @@ def _test():
add_custom_attribute("non_llm_attr", "python-agent")

with pytest.raises(_client_error):
if response_streaming:
stream = bedrock_server.invoke_model_with_response_stream(
body=b"{}",
modelId="does-not-exist",
accept="application/json",
contentType="application/json",
)
for _ in stream:
pass
else:
bedrock_server.invoke_model(
body=b"{}",
modelId="does-not-exist",
accept="application/json",
contentType="application/json",
)
with WithLlmCustomAttributes({"context": "attr"}):
if response_streaming:
stream = bedrock_server.invoke_model_with_response_stream(
body=b"{}",
modelId="does-not-exist",
accept="application/json",
contentType="application/json",
)
for _ in stream:
pass
else:
bedrock_server.invoke_model(
body=b"{}",
modelId="does-not-exist",
accept="application/json",
contentType="application/json",
)

_test()

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,14 +19,15 @@
)
from conftest import BOTOCORE_VERSION # pylint: disable=E0611
from testing_support.fixtures import reset_core_stats_engine, validate_attributes
from testing_support.ml_testing_utils import set_trace_info # noqa: F401
from testing_support.ml_testing_utils import set_trace_info, events_with_context_attrs # noqa: F401
from testing_support.validators.validate_custom_event import validate_custom_event_count
from testing_support.validators.validate_custom_events import validate_custom_events
from testing_support.validators.validate_transaction_metrics import (
validate_transaction_metrics,
)

from newrelic.api.background_task import background_task
from newrelic.api.llm_custom_attributes import WithLlmCustomAttributes
from newrelic.api.transaction import add_custom_attribute

UNSUPPORTED_LANGCHAIN_MODELS = [
Expand Down Expand Up @@ -105,7 +106,7 @@ def test_bedrock_chat_completion_in_txn_with_llm_metadata(
expected_metrics,
response_streaming,
):
@validate_custom_events(expected_events)
@validate_custom_events(events_with_context_attrs(expected_events))
# One summary event, one user message, and one response message from the assistant
@validate_custom_event_count(count=6)
@validate_transaction_metrics(
Expand All @@ -124,6 +125,7 @@ def _test():
add_custom_attribute("llm.conversation_id", "my-awesome-id")
add_custom_attribute("llm.foo", "bar")
add_custom_attribute("non_llm_attr", "python-agent")
exercise_model(prompt="Hi there!")
with WithLlmCustomAttributes({"context": "attr"}):
exercise_model(prompt="Hi there!")

_test()
Loading
Loading