Skip to content

Commit

Permalink
fix(tracing): span start/end time precision (#11484)
Browse files Browse the repository at this point in the history
* fix(tracing): span start/end time precision

Ensure the span lifetime is consistent with the hierarchy
by using the same time source for the root span's end time as the
balancer span.

Before this fix, an approximation (causing a precision error) could
result in the root span to end before its children.

* docs(changelog): add issue

Co-authored-by: Xumin <[email protected]>

---------

Co-authored-by: Xumin <[email protected]>
  • Loading branch information
samugi and StarlightIbuki committed Sep 27, 2023
1 parent b945f17 commit 1a012f3
Show file tree
Hide file tree
Showing 4 changed files with 54 additions and 2 deletions.
9 changes: 9 additions & 0 deletions CHANGELOG/unreleased/kong/11484.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
message: "Tracing: fix an issue that resulted in some parent spans to end before their children due to different precision of their timestamps"
type: bugfix
scope: PDK
prs:
- 11484
jiras:
- "KAG-2336"
issues:
- 11294
2 changes: 2 additions & 0 deletions kong/init.lua
Original file line number Diff line number Diff line change
Expand Up @@ -1467,6 +1467,7 @@ function Kong.log()
local ctx = ngx.ctx
if not ctx.KONG_LOG_START then
ctx.KONG_LOG_START = now() * 1000
ctx.KONG_LOG_START_NS = time_ns()
if is_stream_module then
if not ctx.KONG_PROCESSING_START then
ctx.KONG_PROCESSING_START = start_time() * 1000
Expand Down Expand Up @@ -1535,6 +1536,7 @@ function Kong.log()

if ctx.KONG_BODY_FILTER_START and not ctx.KONG_BODY_FILTER_ENDED_AT then
ctx.KONG_BODY_FILTER_ENDED_AT = ctx.KONG_LOG_START
ctx.KONG_BODY_FILTER_ENDED_AT_NS = ctx.KONG_LOG_START_NS
ctx.KONG_BODY_FILTER_TIME = ctx.KONG_BODY_FILTER_ENDED_AT -
ctx.KONG_BODY_FILTER_START
end
Expand Down
3 changes: 1 addition & 2 deletions kong/tracing/instrumentation.lua
Original file line number Diff line number Diff line change
Expand Up @@ -324,8 +324,7 @@ function _M.runloop_log_before(ctx)
local active_span = tracer.active_span()
-- check root span type to avoid encounter error
if active_span and type(active_span.finish) == "function" then
local end_time = ctx.KONG_BODY_FILTER_ENDED_AT
and ctx.KONG_BODY_FILTER_ENDED_AT * 1e6
local end_time = ctx.KONG_BODY_FILTER_ENDED_AT_NS
active_span:finish(end_time)
end
end
Expand Down
42 changes: 42 additions & 0 deletions spec/02-integration/14-tracing/03-tracer-pdk_spec.lua
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,14 @@ local TCP_PORT = helpers.get_available_port()
local tcp_trace_plugin_name = "tcp-trace-exporter"


local function get_parent(span, spans)
for _, s in ipairs(spans) do
if s.span_id == span.parent_id then
return s
end
end
end

for _, strategy in helpers.each_strategy() do
local proxy_client

Expand Down Expand Up @@ -77,5 +85,39 @@ for _, strategy in helpers.each_strategy() do
end
end)
end)

describe("spans start/end times are consistent with their hierarchy", function ()
lazy_setup(function()
setup_instrumentations("all", false, 1)
end)

lazy_teardown(function()
helpers.stop_kong()
end)

it("sets child lifespan within parent's lifespan", function ()
for _ = 1, 100 do
local thread = helpers.tcp_server(TCP_PORT)
local r = assert(proxy_client:send {
method = "GET",
path = "/",
})
assert.res_status(200, r)

local ok, res = thread:join()
assert.True(ok)
assert.is_string(res)

local spans = cjson.decode(res)
for i = 2, #spans do -- skip the root span (no parent)
local span = spans[i]
local parent = get_parent(span, spans)
assert.is_not_nil(parent)
assert.True(span.start_time_ns >= parent.start_time_ns)
assert.True(span.end_time_ns <= parent.end_time_ns)
end
end
end)
end)
end)
end

0 comments on commit 1a012f3

Please sign in to comment.