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

feat: google-vertexai-instrumentation #289

Merged
merged 35 commits into from
Jan 11, 2024
Merged
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
8f7f36e
feat: added google's vertexai instrumentation
5war00p Dec 25, 2023
fecb37e
fix: lint issues
5war00p Dec 25, 2023
0ef8ceb
format: ci.yml
5war00p Dec 28, 2023
a0c709f
fix: format issues in ci.yml
5war00p Dec 28, 2023
f2f33c2
fix: remove issues
5war00p Dec 28, 2023
96e6e39
fix: formatting issues
5war00p Dec 28, 2023
ffa8419
fix: tabspaces
5war00p Dec 28, 2023
986b652
fix: added top_k attribute
5war00p Dec 29, 2023
e96148f
chore: remove unwanted code and comments
5war00p Dec 29, 2023
a8f8e71
fix: vertextai version specification
5war00p Dec 29, 2023
bb3f654
fix: remove unused imports
5war00p Dec 29, 2023
c25bf4f
feat: chat instrumentation
5war00p Dec 30, 2023
1e3a62c
fix: assert span attribute values
5war00p Dec 31, 2023
7ad0902
Restore: "feat(sdk): user feedback scores" (#254)
nirga Dec 27, 2023
a54ae8b
feat(semconv): added top-k (#291)
nirga Dec 31, 2023
afdd973
feat: chat instrumentation
5war00p Dec 30, 2023
f202bcf
fix: remove print
5war00p Dec 31, 2023
5009e22
Merge branch 'main' into vertexai-instrumentation
5war00p Jan 2, 2024
1989f8e
chore: updated lock file for vertexai project
5war00p Jan 4, 2024
0a48211
feat: added model_name temp solution
5war00p Jan 4, 2024
4082b55
Merge branch 'main' into vertexai-instrumentation
5war00p Jan 4, 2024
85d757e
fix: add all authors
5war00p Jan 5, 2024
ea9603a
fix: test all span attributes
5war00p Jan 5, 2024
a8f5f49
chore: remove google-cloud-aiplatform from pyproject as it unfits fro…
5war00p Jan 8, 2024
74e4c9b
response assertions
5war00p Jan 8, 2024
ea0a570
Merge branch 'main' into vertexai-instrumentation
5war00p Jan 11, 2024
773ca1d
chore: lint and lock
nirga Jan 11, 2024
88605e0
fix: lock file
nirga Jan 11, 2024
46c6927
chore: vertex auth
nirga Jan 11, 2024
51b14cd
Merge branch 'main' into pr/289
nirga Jan 11, 2024
c4ecaa5
Merge branch 'main' into pr/289
nirga Jan 11, 2024
92ab5c4
fix: excluded URLs for vertex AI
nirga Jan 11, 2024
40f2a3e
fix: async wrapper missing
5war00p Jan 11, 2024
3377e20
fix: lint
5war00p Jan 11, 2024
6fd3a54
fix: sample app
nirga Jan 11, 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 .github/workflows/ci.yml
5war00p marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,8 @@ jobs:
PINECONE_API_KEY: ${{ secrets.PINECONE_API_KEY }}
PINECONE_ENVIRONMENT: ${{ secrets.PINECONE_ENVIRONMENT }}
TRACELOOP_API_KEY: ${{ secrets.TRACELOOP_API_KEY }}
VERTEXAI_PROJECT_ID: ${{ secrets.VERTEXAI_PROJECT_ID }}
VERTEXAI_LOCATION: ${{ secrets.VERTEXAI_LOCATION }}
run: npx nx affected -t test --exclude='sample-app' --parallel=3

build-sample-app:
Expand Down
26 changes: 24 additions & 2 deletions packages/opentelemetry-instrumentation-vertexai/README.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,25 @@
# opentelemetry-instrumentation-vertexai
# OpenTelemetry VertexAI Instrumentation

Project description here.
<a href="https://pypi.org/project/opentelemetry-instrumentation-vertexai/">
<img src="https://badge.fury.io/py/opentelemetry-instrumentation-vertexai.svg">
</a>

This library allows tracing VertexAI prompts and completions sent with the official [VertexAI library](https://github.com/googleapis/python-aiplatform).

## Installation

```bash
pip install opentelemetry-instrumentation-vertexai
```

## Privacy

**By default, this instrumentation logs prompts, completions, and embeddings to span attributes**. This gives you a clear visibility into how your LLM application is working, and can make it easy to debug and evaluate the quality of the outputs.

However, you may want to disable this logging for privacy reasons, as they may contain highly sensitive data from your users. You may also simply want to reduce the size of your traces.

To disable logging, set the `TRACELOOP_TRACE_CONTENT` environment variable to `false`.

```bash
TRACELOOP_TRACE_CONTENT=false
```
Original file line number Diff line number Diff line change
@@ -1 +1,299 @@
"""OpenTelemetry Vertex AI instrumentation"""
import logging
import os
import types
from typing import Collection
from wrapt import wrap_function_wrapper

from opentelemetry import context as context_api
from opentelemetry.trace import get_tracer, SpanKind
from opentelemetry.trace.status import Status, StatusCode

from opentelemetry.instrumentation.instrumentor import BaseInstrumentor
from opentelemetry.instrumentation.utils import (
_SUPPRESS_INSTRUMENTATION_KEY,
unwrap
)

from opentelemetry.semconv.ai import SpanAttributes, LLMRequestTypeValues
from opentelemetry.instrumentation.vertexai.version import __version__

logger = logging.getLogger(__name__)

_instruments = ("google-cloud-aiplatform >= 1.38.1",)

llm_model = 'unknown'

WRAPPED_METHODS = [
{
"package": "vertexai.preview.generative_models",
"object": "GenerativeModel",
"method": "__init__",
"span_name": "vertexai.__init__",
},
{
"package": "vertexai.preview.generative_models",
"object": "GenerativeModel",
"method": "generate_content",
"span_name": "vertexai.generate_content",
},
{
"package": "vertexai.language_models",
"object": "TextGenerationModel",
"method": "from_pretrained",
"span_name": "vertexai.from_pretrained",
},
{
"package": "vertexai.language_models",
"object": "TextGenerationModel",
"method": "predict",
"span_name": "vertexai.predict",
},
{
"package": "vertexai.language_models",
"object": "TextGenerationModel",
"method": "predict_async",
"span_name": "vertexai.predict",
},
{
"package": "vertexai.language_models",
"object": "TextGenerationModel",
"method": "predict_streaming",
"span_name": "vertexai.predict",
},
{
"package": "vertexai.language_models",
"object": "TextGenerationModel",
"method": "predict_streaming_async",
"span_name": "vertexai.predict",
},
{
"package": "vertexai.language_models",
"object": "ChatModel",
"method": "from_pretrained",
"span_name": "vertexai.from_pretrained",
},
{
"package": "vertexai.language_models",
"object": "ChatSession",
"method": "send_message",
"span_name": "vertexai.send_message",
},
{
"package": "vertexai.language_models",
"object": "ChatSession",
"method": "send_message_streaming",
"span_name": "vertexai.send_message",
},
]


def should_send_prompts():
return (
os.getenv("TRACELOOP_TRACE_CONTENT") or "true"
).lower() == "true" or context_api.get_value("override_enable_content_tracing")

def is_streaming_response(response):
return isinstance(response, types.GeneratorType)

def is_async_streaming_response(response):
return isinstance(response, types.AsyncGeneratorType)

def _set_span_attribute(span, name, value):
if value is not None:
if value != "":
span.set_attribute(name, value)
return

def _set_input_attributes(span, args, kwargs):

if (should_send_prompts() and args is not None and len(args) > 0):
prompt = ''
for arg in args:
if isinstance(arg, str):
prompt = f"{prompt}{arg}\n"
elif isinstance(arg, list):
for subarg in arg:
prompt = f"{prompt}{subarg}\n"

_set_span_attribute(
span,
f"{SpanAttributes.LLM_PROMPTS}.0.user",
prompt,
)

_set_span_attribute(span, SpanAttributes.LLM_REQUEST_MODEL, llm_model)
_set_span_attribute(span, f"{SpanAttributes.LLM_PROMPTS}.0.user", kwargs.get('prompt'))
_set_span_attribute(span, SpanAttributes.LLM_TEMPERATURE, kwargs.get('temperature'))
5war00p marked this conversation as resolved.
Show resolved Hide resolved
_set_span_attribute(span, SpanAttributes.LLM_REQUEST_MAX_TOKENS, kwargs.get('max_output_tokens'))
_set_span_attribute(span, SpanAttributes.LLM_TOP_P, kwargs.get('top_p'))
5war00p marked this conversation as resolved.
Show resolved Hide resolved
_set_span_attribute(span, SpanAttributes.LLM_TOP_K, kwargs.get('top_k'))
_set_span_attribute(span, SpanAttributes.LLM_PRESENCE_PENALTY, kwargs.get('presence_penalty'))
_set_span_attribute(span, SpanAttributes.LLM_FREQUENCY_PENALTY, kwargs.get('frequency_penalty'))

return

def _set_response_attributes(span, response):
_set_span_attribute(span, SpanAttributes.LLM_RESPONSE_MODEL, llm_model)

if hasattr(response, 'text'):
if hasattr(response, '_raw_response') and hasattr(response._raw_response, 'usage_metadata'):
_set_span_attribute(
span, SpanAttributes.LLM_USAGE_TOTAL_TOKENS, response._raw_response.usage_metadata.total_token_count
)
_set_span_attribute(
span, SpanAttributes.LLM_USAGE_COMPLETION_TOKENS, response._raw_response.usage_metadata.candidates_token_count
)
_set_span_attribute(
span, SpanAttributes.LLM_USAGE_PROMPT_TOKENS, response._raw_response.usage_metadata.prompt_token_count
)


if isinstance(response.text, list):
for index, item in enumerate(response):
prefix = f"{SpanAttributes.LLM_COMPLETIONS}.{index}"
_set_span_attribute(span, f"{prefix}.content", item.text)
elif isinstance(response.text, str):
_set_span_attribute(
span, f"{SpanAttributes.LLM_COMPLETIONS}.0.content", response.text
)
else:
if isinstance(response, list):
for index, item in enumerate(response):
prefix = f"{SpanAttributes.LLM_COMPLETIONS}.{index}"
_set_span_attribute(span, f"{prefix}.content", item)
elif isinstance(response, str):
_set_span_attribute(
span, f"{SpanAttributes.LLM_COMPLETIONS}.0.content", response
)

return

def _build_from_streaming_response(span, response):
complete_response = ""
for item in response:
item_to_yield = item
complete_response += str(item)

yield item_to_yield

_set_response_attributes(span, complete_response)

span.set_status(Status(StatusCode.OK))
span.end()

async def _abuild_from_streaming_response(span, response):
complete_response = ""
async for item in response:
item_to_yield = item
complete_response += str(item)

yield item_to_yield

_set_response_attributes(span, complete_response)

span.set_status(Status(StatusCode.OK))
span.end()

def _handle_request(span, args, kwargs):
try:
if span.is_recording():
_set_input_attributes(span, args, kwargs)

except Exception as ex: # pylint: disable=broad-except
logger.warning(
"Failed to set input attributes for VertexAI span, error: %s", str(ex)
)

def _handle_response(span, response):
try:
if span.is_recording():
_set_response_attributes(span, response)

except Exception as ex: # pylint: disable=broad-except
logger.warning(
"Failed to set response attributes for VertexAI span, error: %s",
str(ex),
)
if span.is_recording():
span.set_status(Status(StatusCode.OK))


def _with_tracer_wrapper(func):
"""Helper for providing tracer for wrapper functions."""

def _with_tracer(tracer, to_wrap):
def wrapper(wrapped, instance, args, kwargs):
return func(tracer, to_wrap, wrapped, instance, args, kwargs)

return wrapper

return _with_tracer

@_with_tracer_wrapper
def _wrap(tracer, to_wrap, wrapped, instance, args, kwargs):

"""Instruments and calls every function defined in TO_WRAP."""
if context_api.get_value(_SUPPRESS_INSTRUMENTATION_KEY):
return wrapped(*args, **kwargs)

global llm_model

if (to_wrap.get('method') == 'from_pretrained' or to_wrap.get('method') == "__init__") and args is not None and len(args) > 0:
llm_model = args[0]
return wrapped(*args, **kwargs)

name = to_wrap.get("span_name")
span = tracer.start_span(
name,
kind=SpanKind.CLIENT,
attributes={
SpanAttributes.LLM_VENDOR: "VertexAI",
SpanAttributes.LLM_REQUEST_TYPE: LLMRequestTypeValues.COMPLETION.value,
},
)

_handle_request(span, args, kwargs)

response = wrapped(*args, **kwargs)

if response:
if is_streaming_response(response):
return _build_from_streaming_response(span, response)
elif is_async_streaming_response(response):
return _abuild_from_streaming_response(span, response)
else:
_handle_response(span, response)

span.end()
return response


class VertexAIInstrumentor(BaseInstrumentor):
"""An instrumentor for VertextAI's client library."""

def instrumentation_dependencies(self) -> Collection[str]:
return _instruments

def _instrument(self, **kwargs):
tracer_provider = kwargs.get("tracer_provider")
tracer = get_tracer(__name__, __version__, tracer_provider)
for wrapped_method in WRAPPED_METHODS:
wrap_package = wrapped_method.get("package")
wrap_object = wrapped_method.get("object")
wrap_method = wrapped_method.get("method")

wrap_function_wrapper(
wrap_package,
f"{wrap_object}.{wrap_method}",
_wrap(tracer, wrapped_method),
)

def _uninstrument(self, **kwargs):
for wrapped_method in WRAPPED_METHODS:
wrap_package = wrapped_method.get("package")
wrap_object = wrapped_method.get("object")
unwrap(
f"{wrap_package}.{wrap_object}",
wrapped_method.get("method", ""),
)
Loading
Loading