15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
|
# File 'lib/dspy/context.rb', line 15
def with_span(operation:, **attributes)
span_id = SecureRandom.uuid
parent_span_id = current[:span_stack].last
start_time = Process.clock_gettime(Process::CLOCK_MONOTONIC)
span_attributes = {
trace_id: current[:trace_id],
span_id: span_id,
parent_span_id: parent_span_id,
operation: operation,
**attributes
}
DSPy.log('span.start', **span_attributes)
current[:span_stack].push(span_id)
begin
if DSPy::Observability.enabled? && DSPy::Observability.tracer
span_attributes = attributes.transform_keys(&:to_s).reject { |k, v| v.nil? }
if current[:span_stack].length == 1 span_attributes['langfuse.trace.name'] = operation
end
otel_start_time = Time.now
DSPy::Observability.tracer.in_span(
operation,
attributes: span_attributes,
kind: :internal
) do |span|
result = yield(span)
if span
duration_ms = ((Time.now - otel_start_time) * 1000).round(3)
span.set_attribute('duration.ms', duration_ms)
span.set_attribute('langfuse.observation.startTime', otel_start_time.iso8601(3))
span.set_attribute('langfuse.observation.endTime', Time.now.iso8601(3))
end
result
end
else
yield(nil)
end
ensure
current[:span_stack].pop
duration_ms = ((Process.clock_gettime(Process::CLOCK_MONOTONIC) - start_time) * 1000).round(2)
DSPy.log('span.end',
trace_id: current[:trace_id],
span_id: span_id,
duration_ms: duration_ms
)
end
end
|