scale-gp-beta 0.1.0a26__py3-none-any.whl → 0.1.0a27__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
scale_gp_beta/_version.py CHANGED
@@ -1,4 +1,4 @@
1
1
  # File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
2
2
 
3
3
  __title__ = "scale_gp_beta"
4
- __version__ = "0.1.0-alpha.26" # x-release-please-version
4
+ __version__ = "0.1.0-alpha.27" # x-release-please-version
@@ -0,0 +1,5 @@
1
+ try:
2
+ from .openai.openai_tracing_sgp_processor import OpenAITracingSGPProcessor # type: ignore[attr-defined]
3
+ __all__ = ["OpenAITracingSGPProcessor"]
4
+ except ImportError:
5
+ pass
@@ -0,0 +1,29 @@
1
+ from typing import Dict, Optional
2
+
3
+ from scale_gp_beta.lib.tracing.types import SpanTypeLiterals
4
+
5
+ OPENAI_SPAN_TYPE_MAP: Dict[str, SpanTypeLiterals] = {
6
+ "generation": "COMPLETION",
7
+ "agent": "AGENT_WORKFLOW",
8
+ "function": "CODE_EXECUTION",
9
+ "response": "COMPLETION",
10
+ "handoff": "CUSTOM",
11
+ "custom": "CUSTOM",
12
+ "guardrail": "STANDALONE",
13
+ "transcription": "STANDALONE",
14
+ "speech": "STANDALONE",
15
+ "speech_group": "STANDALONE",
16
+ "mcp_tools": "STANDALONE",
17
+ }
18
+
19
+
20
+ def openai_span_type_map(span_type: Optional[str]) -> SpanTypeLiterals:
21
+ """
22
+ Maps an OpenAI span type string to its corresponding SGP SpanTypeLiteral.
23
+
24
+ Returns "CUSTOM" if the input is None or not found in the map.
25
+ """
26
+ if not span_type:
27
+ return "CUSTOM"
28
+
29
+ return OPENAI_SPAN_TYPE_MAP.get(span_type, "CUSTOM")
@@ -0,0 +1,144 @@
1
+ # The following `type: ignore` is necessary because this file provides an optional
2
+ # integration for the 'openai-agents' library. The linter will report missing
3
+ # imports if the library isn't installed in the development environment, but the
4
+ # code handles this gracefully at runtime with a try/except block.
5
+ # type: ignore
6
+ import logging
7
+ from typing import TYPE_CHECKING, Any, Dict, Optional
8
+
9
+ from scale_gp_beta.lib.tracing.span import BaseSpan
10
+ from scale_gp_beta.lib.tracing.tracing import create_span, flush_queue, create_trace, tracing_queue_manager
11
+ from scale_gp_beta.lib.tracing.trace_queue_manager import TraceQueueManager
12
+
13
+ from .utils import sgp_span_name, parse_metadata, extract_response, parse_input_output
14
+ from .openai_span_type_map import openai_span_type_map
15
+
16
+ log: logging.Logger = logging.getLogger(__name__)
17
+
18
+ try:
19
+ from agents.tracing import (
20
+ Span as OpenAISpan,
21
+ Trace as OpenAITrace,
22
+ SpanData as OpenAISpanData,
23
+ ResponseSpanData as OpenAIResponseSpanData,
24
+ TracingProcessor,
25
+ )
26
+
27
+ _openai_agents_installed = True
28
+ except ImportError:
29
+ _openai_agents_installed = False
30
+ TracingProcessor = object
31
+ class _Fallback:
32
+ pass
33
+
34
+ OpenAISpan = _Fallback
35
+ OpenAITrace = _Fallback
36
+ OpenAISpanData = _Fallback
37
+ OpenAIResponseSpanData = _Fallback
38
+
39
+ if TYPE_CHECKING:
40
+ OpenAISpan = Any
41
+ OpenAITrace = Any
42
+ OpenAISpanData = Any
43
+ OpenAIResponseSpanData = Any
44
+
45
+
46
+ class OpenAITracingSGPProcessor(TracingProcessor):
47
+ """
48
+ An SGP tracing processor that integrates with OpenAI Agents SDK.
49
+
50
+ This processor requires the 'openai-agents' library to be installed.
51
+ It will raise an ImportError on initialization if the library is not found.
52
+ """
53
+
54
+ def __init__(
55
+ self,
56
+ *,
57
+ queue_manager: Optional[TraceQueueManager] = None,
58
+ ):
59
+ """
60
+ Initializes the processor.
61
+
62
+ Args:
63
+ queue_manager: An optional custom trace queue manager.
64
+
65
+ Raises:
66
+ ImportError: If the 'openai-agents' package is not installed.
67
+ """
68
+ # This runtime check is the gatekeeper. If 'openai-agents' isn't
69
+ # installed, no instance of this class can be created.
70
+ if not _openai_agents_installed:
71
+ raise ImportError(
72
+ "The 'openai-agents' package is required to use the OpenAITracingSGPProcessor. "
73
+ "Please install it with: pip install 'openai-agents'"
74
+ )
75
+
76
+ self._queue_manager = queue_manager or tracing_queue_manager()
77
+ self._spans: Dict[str, BaseSpan] = {}
78
+
79
+ def on_trace_start(self, openai_trace: "OpenAITrace") -> None:
80
+ sgp_trace = create_trace(
81
+ name=openai_trace.name or "Agent Workflow",
82
+ span_type="AGENT_WORKFLOW",
83
+ group_id=openai_trace.group_id,
84
+ trace_id=openai_trace.trace_id,
85
+ metadata={"openai_metadata": openai_trace.metadata or {}},
86
+ )
87
+ sgp_trace.start()
88
+ self._spans[openai_trace.trace_id] = sgp_trace.root_span
89
+
90
+ def on_trace_end(self, openai_trace: "OpenAITrace") -> None:
91
+ root_span = self._spans.pop(openai_trace.trace_id, None)
92
+ if root_span is None:
93
+ log.warning(f"No root span found for trace_id: {openai_trace.trace_id}")
94
+ return
95
+ root_span.end()
96
+
97
+ def on_span_start(self, openai_span: "OpenAISpan") -> None:
98
+ _spans_key = openai_span.parent_id or openai_span.trace_id
99
+ parent_span = self._spans.get(_spans_key)
100
+ span_data = openai_span.span_data
101
+
102
+ if not parent_span:
103
+ log.warning(f"No parent span found for span_id: {openai_span.span_id}")
104
+ return
105
+
106
+ name = sgp_span_name(openai_span)
107
+ sgp_span_type = openai_span_type_map(span_data.type if span_data else None)
108
+
109
+ new_span = create_span(
110
+ name=name,
111
+ span_type=sgp_span_type,
112
+ span_id=openai_span.span_id,
113
+ trace_id=parent_span.trace_id,
114
+ parent_id=parent_span.span_id,
115
+ )
116
+ new_span.start()
117
+ self._spans[openai_span.span_id] = new_span
118
+
119
+ def on_span_end(self, openai_span: "OpenAISpan") -> None:
120
+ sgp_span = self._spans.pop(openai_span.span_id, None)
121
+ if sgp_span is None:
122
+ log.warning(f"No existing span found for span_id: {openai_span.span_id}")
123
+ return
124
+
125
+ # special case response type, if there are more like this we should change arch to support this
126
+ if isinstance(openai_span.span_data, OpenAIResponseSpanData):
127
+ response = extract_response(openai_span)
128
+ sgp_span.input = {"input": openai_span.span_data.input}
129
+ sgp_span.output = {"output": response.get("output", None) if response else None }
130
+ else:
131
+ sgp_span.output = parse_input_output(openai_span, "output")
132
+ sgp_span.input = parse_input_output(openai_span, "input")
133
+
134
+ sgp_span.metadata = parse_metadata(openai_span)
135
+
136
+ if openai_span.error:
137
+ sgp_span.set_error(error_message=str(openai_span.error))
138
+ sgp_span.end()
139
+
140
+ def shutdown(self) -> None:
141
+ self._queue_manager.shutdown()
142
+
143
+ def force_flush(self) -> None:
144
+ flush_queue()
@@ -0,0 +1,68 @@
1
+ # type: ignore
2
+ import json
3
+ from typing import Any, Dict, Optional
4
+
5
+ try:
6
+ from agents.tracing import (
7
+ Span as OpenAISpan,
8
+ ResponseSpanData as OpenAIResponseSpanData,
9
+ )
10
+ except ImportError:
11
+ OpenAISpan = Any
12
+ OpenAIResponseSpanData = Any
13
+
14
+
15
+ def sgp_span_name(openai_span: "OpenAISpan") -> str:
16
+ span_data = openai_span.span_data
17
+ if span_data is None:
18
+ return "Unnamed Span"
19
+ default_name = str(type(span_data).__name__).replace("SpanData", "")
20
+
21
+ return span_data.export().get("name", default_name)
22
+
23
+
24
+ def extract_response(openai_span: "OpenAISpan") -> Optional[Dict[str, Any]]:
25
+ span_data = openai_span.span_data
26
+ if not isinstance(span_data, OpenAIResponseSpanData):
27
+ return None
28
+
29
+ return {} if span_data.response is None else span_data.response.model_dump()
30
+
31
+
32
+ def parse_metadata(openai_span: "OpenAISpan") -> Dict[str, Any]:
33
+ """Extracts and formats metadata from an OpenAI span."""
34
+ if not openai_span.span_data:
35
+ return {}
36
+
37
+ excluded_fields = ("output", "input")
38
+ exported_span_data: dict = openai_span.span_data.export()
39
+ exported_span_data["openai_span_id"] = openai_span.span_id
40
+
41
+ filtered_span_data: dict = {
42
+ k: v for k, v in exported_span_data.items() if k not in excluded_fields
43
+ }
44
+ response = extract_response(openai_span)
45
+ if response is not None:
46
+ filtered_span_data["openai_response"] = response
47
+ return {"openai_metadata": filtered_span_data}
48
+
49
+
50
+ def parse_input_output(openai_span: "OpenAISpan", key: str) -> Dict[str, Any]:
51
+ """
52
+ Safely extracts and parses input/output data from a span.
53
+ The openai-agents SDK sometimes returns this data as a JSON string.
54
+ """
55
+ span_data = openai_span.span_data
56
+ if not span_data:
57
+ return {}
58
+ exported_span_data: dict = span_data.export()
59
+ value: Optional[Any] = exported_span_data.get(key)
60
+ if value is None:
61
+ return {}
62
+
63
+ if isinstance(value, str):
64
+ try:
65
+ return json.loads(value)
66
+ except (json.JSONDecodeError, TypeError):
67
+ return {key: value}
68
+ return {key: value}
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: scale-gp-beta
3
- Version: 0.1.0a26
3
+ Version: 0.1.0a27
4
4
  Summary: The official Python library for the Scale GP API
5
5
  Project-URL: Homepage, https://github.com/scaleapi/sgp-python-beta
6
6
  Project-URL: Repository, https://github.com/scaleapi/sgp-python-beta
@@ -11,7 +11,7 @@ scale_gp_beta/_resource.py,sha256=siZly_U6D0AOVLAzaOsqUdEFFzVMbWRj-ml30nvRp7E,11
11
11
  scale_gp_beta/_response.py,sha256=GemuybPk0uemovTlGHyHkj-ScYTTDJA0jqH5FQqIPwQ,28852
12
12
  scale_gp_beta/_streaming.py,sha256=fcCSGXslmi2SmmkM05g2SACXHk2Mj7k1X5uMBu6U5s8,10112
13
13
  scale_gp_beta/_types.py,sha256=0wSs40TefKMPBj2wQKenEeZ0lzedoHClNJeqrpAgkII,6204
14
- scale_gp_beta/_version.py,sha256=a3A3ZJ3GGbEmFnL4rjXbroDX1dq8r9ZDEI2rLL3BuIU,174
14
+ scale_gp_beta/_version.py,sha256=pH9t8LZndiar4fooop-BkuK6f4yzE0eHL7DxaCUG-24,174
15
15
  scale_gp_beta/pagination.py,sha256=t-U86PYxl20VRsz8VXOMJJDe7HxkX7ISFMvRNbBNy9s,4054
16
16
  scale_gp_beta/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
17
17
  scale_gp_beta/_utils/__init__.py,sha256=PNZ_QJuzZEgyYXqkO1HVhGkj5IU9bglVUcw7H-Knjzw,2062
@@ -35,6 +35,11 @@ scale_gp_beta/lib/tracing/trace_queue_manager.py,sha256=RnEDOKi7c1xlKXUsofShzZ8W
35
35
  scale_gp_beta/lib/tracing/tracing.py,sha256=_j30MMy-w5Y33t2lAYSFVpMY_Ia2PFcEFaUEG5UN2OI,9368
36
36
  scale_gp_beta/lib/tracing/types.py,sha256=07piipdyN0a25gYGtV6dM12ZroJqLUpBmcFmdQPV4jU,1113
37
37
  scale_gp_beta/lib/tracing/util.py,sha256=8Oq4wLXRNOzh3CC1zRaBEr0h_WdXLrk536BUNKRddVE,1527
38
+ scale_gp_beta/lib/tracing/integrations/__init__.py,sha256=-1CwzEpjZS0eifPRAqwEA4D4nzEAfEe9WbXrLxCWQvg,187
39
+ scale_gp_beta/lib/tracing/integrations/openai/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
40
+ scale_gp_beta/lib/tracing/integrations/openai/openai_span_type_map.py,sha256=kjEVpTJBIvyxNeBbc8k2MBI6-mdIOAPahyk8nAT7HEQ,828
41
+ scale_gp_beta/lib/tracing/integrations/openai/openai_tracing_sgp_processor.py,sha256=o-Fp_V7U9cC1kKP0_750GQInKGlF7KAdpIbGTmyu2tg,5270
42
+ scale_gp_beta/lib/tracing/integrations/openai/utils.py,sha256=s6tbNFJ3N7GjqXDM9f2f4r9ugz3wjO0OTU6KzlU_tLQ,2113
38
43
  scale_gp_beta/resources/__init__.py,sha256=uquYxoMtAXyfIas331lnYexJZKr7pEYqi_vKlfrHRIg,5431
39
44
  scale_gp_beta/resources/completions.py,sha256=4esj9lGTJAxt6wFvON126DvEGkMIChRZ6uZBOf56Aac,31868
40
45
  scale_gp_beta/resources/dataset_items.py,sha256=2d7O5zmqVEafJTxVwgbRz9yq-4T81dPPfFuPDRAaWqU,22510
@@ -121,7 +126,7 @@ scale_gp_beta/types/chat/completion_models_params.py,sha256=NdZ3Lkgclq6UKc9iae-e
121
126
  scale_gp_beta/types/chat/completion_models_response.py,sha256=Ctgj6o-QWPSdjBKzG9J4Id0-DjXu4UGGw1NR6-840Ec,403
122
127
  scale_gp_beta/types/chat/model_definition.py,sha256=XPkHEe4tY442oLSMlaotClNDv5XeQ3j5RrXgLZFo1Qk,841
123
128
  scale_gp_beta/types/files/__init__.py,sha256=OKfJYcKb4NObdiRObqJV_dOyDQ8feXekDUge2o_4pXQ,122
124
- scale_gp_beta-0.1.0a26.dist-info/METADATA,sha256=prd8poY95iVXq74tFDvSORgl6xx0wENE4p7SBf1Po-w,28723
125
- scale_gp_beta-0.1.0a26.dist-info/WHEEL,sha256=C2FUgwZgiLbznR-k0b_5k3Ai_1aASOXDss3lzCUsUug,87
126
- scale_gp_beta-0.1.0a26.dist-info/licenses/LICENSE,sha256=x49Bj8r_ZpqfzThbmfHyZ_bE88XvHdIMI_ANyLHFFRE,11338
127
- scale_gp_beta-0.1.0a26.dist-info/RECORD,,
129
+ scale_gp_beta-0.1.0a27.dist-info/METADATA,sha256=BXtYGBjQHmzC9ciL8a1S5IW29yawXZiefPmJfbMlY3U,28723
130
+ scale_gp_beta-0.1.0a27.dist-info/WHEEL,sha256=C2FUgwZgiLbznR-k0b_5k3Ai_1aASOXDss3lzCUsUug,87
131
+ scale_gp_beta-0.1.0a27.dist-info/licenses/LICENSE,sha256=x49Bj8r_ZpqfzThbmfHyZ_bE88XvHdIMI_ANyLHFFRE,11338
132
+ scale_gp_beta-0.1.0a27.dist-info/RECORD,,