From adbf87767f6d9da92119f3eb128db9b3e42a4973 Mon Sep 17 00:00:00 2001 From: Reasno Date: Thu, 5 Nov 2020 12:15:17 +0800 Subject: [PATCH] fix: safely finish spans in case of exception also make tracing aspects usable in CLI mode. --- src/Aspect/HttpClientAspect.php | 12 ++++++++---- src/Aspect/JsonRpcAspect.php | 15 +++++++++------ src/Aspect/MethodAspect.php | 7 +++++-- src/Aspect/RedisAspect.php | 9 ++++++--- src/Aspect/TraceAnnotationAspect.php | 7 +++++-- src/SpanStarter.php | 7 ++++++- 6 files changed, 39 insertions(+), 18 deletions(-) diff --git a/src/Aspect/HttpClientAspect.php b/src/Aspect/HttpClientAspect.php index a3c67b5..805f1de 100644 --- a/src/Aspect/HttpClientAspect.php +++ b/src/Aspect/HttpClientAspect.php @@ -91,11 +91,15 @@ public function process(ProceedingJoinPoint $proceedingJoinPoint) ); $options['headers'] = array_replace($options['headers'] ?? [], $appendHeaders); $proceedingJoinPoint->arguments['keys']['options'] = $options; - $result = $proceedingJoinPoint->process(); - if ($result instanceof ResponseInterface) { - $span->setTag($this->spanTagManager->get('http_client', 'http.status_code'), $result->getStatusCode()); + + try { + $result = $proceedingJoinPoint->process(); + if ($result instanceof ResponseInterface) { + $span->setTag($this->spanTagManager->get('http_client', 'http.status_code'), $result->getStatusCode()); + } + } finally { + $span->finish(); } - $span->finish(); return $result; } } diff --git a/src/Aspect/JsonRpcAspect.php b/src/Aspect/JsonRpcAspect.php index 9a28ed4..218e5da 100644 --- a/src/Aspect/JsonRpcAspect.php +++ b/src/Aspect/JsonRpcAspect.php @@ -90,13 +90,16 @@ public function process(ProceedingJoinPoint $proceedingJoinPoint) } if ($proceedingJoinPoint->methodName === 'send') { - $result = $proceedingJoinPoint->process(); - /** @var Span $span */ - if ($span = CT::get('tracer.span.' . static::class)) { - if ($this->spanTagManager->has('rpc', 'status')) { - $span->setTag($this->spanTagManager->get('rpc', 'status'), isset($result['result']) ? 'OK' : 'Failed'); + try { + $result = $proceedingJoinPoint->process(); + } finally { + /** @var Span $span */ + if ($span = CT::get('tracer.span.' . static::class)) { + if ($this->spanTagManager->has('rpc', 'status')) { + $span->setTag($this->spanTagManager->get('rpc', 'status'), isset($result['result']) ? 'OK' : 'Failed'); + } + $span->finish(); } - $span->finish(); } return $result; diff --git a/src/Aspect/MethodAspect.php b/src/Aspect/MethodAspect.php index d51fa83..18df9f8 100644 --- a/src/Aspect/MethodAspect.php +++ b/src/Aspect/MethodAspect.php @@ -58,8 +58,11 @@ public function process(ProceedingJoinPoint $proceedingJoinPoint) $key = $proceedingJoinPoint->className . '::' . $proceedingJoinPoint->methodName; $span = $this->startSpan($key); - $result = $proceedingJoinPoint->process(); - $span->finish(); + try { + $result = $proceedingJoinPoint->process(); + } finally { + $span->finish(); + } return $result; } } diff --git a/src/Aspect/RedisAspect.php b/src/Aspect/RedisAspect.php index 5144221..3c22714 100644 --- a/src/Aspect/RedisAspect.php +++ b/src/Aspect/RedisAspect.php @@ -74,9 +74,12 @@ public function process(ProceedingJoinPoint $proceedingJoinPoint) $arguments = $proceedingJoinPoint->arguments['keys']; $span = $this->startSpan('Redis' . '::' . $arguments['name']); $span->setTag($this->spanTagManager->get('redis', 'arguments'), json_encode($arguments['arguments'])); - $result = $proceedingJoinPoint->process(); - $span->setTag($this->spanTagManager->get('redis', 'result'), json_encode($result)); - $span->finish(); + try { + $result = $proceedingJoinPoint->process(); + $span->setTag($this->spanTagManager->get('redis', 'result'), json_encode($result)); + } finally { + $span->finish(); + } return $result; } } diff --git a/src/Aspect/TraceAnnotationAspect.php b/src/Aspect/TraceAnnotationAspect.php index fd9eee6..46ea182 100644 --- a/src/Aspect/TraceAnnotationAspect.php +++ b/src/Aspect/TraceAnnotationAspect.php @@ -58,8 +58,11 @@ public function process(ProceedingJoinPoint $proceedingJoinPoint) } $span = $this->startSpan($name); $span->setTag($tag, $source); - $result = $proceedingJoinPoint->process(); - $span->finish(); + try { + $result = $proceedingJoinPoint->process(); + } finally { + $span->finish(); + } return $result; } } diff --git a/src/SpanStarter.php b/src/SpanStarter.php index f22cf6c..68ea131 100644 --- a/src/SpanStarter.php +++ b/src/SpanStarter.php @@ -36,7 +36,12 @@ protected function startSpan( /** @var ServerRequestInterface $request */ $request = Context::get(ServerRequestInterface::class); if (! $request instanceof ServerRequestInterface) { - throw new \RuntimeException('ServerRequest object missing.'); + // If the request object is absent, we are probably in a commandline context. + // Throwing an exception is unnecessary. + $root = $this->tracer->startSpan($name, $option); + $root->setTag(SPAN_KIND, $kind); + Context::set('tracer.root', $root); + return $root; } $carrier = array_map(function ($header) { return $header[0];