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

Add a rich console exporter #686

Merged
merged 15 commits into from
Sep 29, 2021
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
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
28 changes: 28 additions & 0 deletions exporter/opentelemetry-exporter-richconsole/README.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
OpenTelemetry Rich Console Exporter
===================================

|pypi|

.. |pypi| image:: https://badge.fury.io/py/opentelemetry-exporter-richconsole.svg
:target: https://pypi.org/project/opentelemetry-exporter-richconsole/

This library is a console exporter using the Rich tree view. When used with a batch span processor, the rich console exporter will show the trace as a
tree and all related spans as children within the tree, including properties.

Installation
------------

::

pip install opentelemetry-exporter-richconsole
tonybaloney marked this conversation as resolved.
Show resolved Hide resolved


.. _Rich: https://rich.readthedocs.io/
.. _OpenTelemetry: https://github.com/open-telemetry/opentelemetry-python/


References
----------

* `Rich <https://rich.readthedocs.io/>`_
* `OpenTelemetry Project <https://opentelemetry.io/>`_
50 changes: 50 additions & 0 deletions exporter/opentelemetry-exporter-richconsole/setup.cfg
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
# Copyright The OpenTelemetry Authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
[metadata]
name = opentelemetry-exporter-richconsole
description = Rich Console Exporter for OpenTelemetry
long_description = file: README.rst
long_description_content_type = text/x-rst
author = OpenTelemetry Authors
author_email = [email protected]
url = https://github.com/open-telemetry/opentelemetry-python-contrib/exporter/opentelemetry-exporter-richconsole
platforms = any
license = Apache-2.0
classifiers =
Development Status :: 4 - Beta
Intended Audience :: Developers
License :: OSI Approved :: Apache Software License
Programming Language :: Python
Programming Language :: Python :: 3
Programming Language :: Python :: 3.6
Programming Language :: Python :: 3.7
Programming Language :: Python :: 3.8
tonybaloney marked this conversation as resolved.
Show resolved Hide resolved

[options]
python_requires = >=3.6
package_dir=
=src
packages=find_namespace:
install_requires =
rich>=10.0.0
opentelemetry-api ~= 1.3
opentelemetry-sdk ~= 1.3
opentelemetry-semantic-conventions == 0.24b0

[options.packages.find]
where = src

[options.extras_require]
test =
27 changes: 27 additions & 0 deletions exporter/opentelemetry-exporter-richconsole/setup.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
# Copyright The OpenTelemetry Authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import os

import setuptools

BASE_DIR = os.path.dirname(__file__)
VERSION_FILENAME = os.path.join(
BASE_DIR, "src", "opentelemetry", "exporter", "richconsole", "version.py"
)
PACKAGE_INFO = {}
with open(VERSION_FILENAME) as f:
exec(f.read(), PACKAGE_INFO)

setuptools.setup(version=PACKAGE_INFO["__version__"])
Original file line number Diff line number Diff line change
@@ -0,0 +1,171 @@
# Copyright The OpenTelemetry Authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

"""
The **OpenTelemetry Rich Console Exporter** provides a span exporter from a batch span processor
to print `OpenTelemetry`_ traces using `Rich`_.

Installation
------------

::

pip install opentelemetry-exporter-richconsole


Usage
-----

The Rich Console Exporter is a console exporter that prints a tree view onto stdout of the traces
with the related spans and properties as children of that tree. The Rich Console Exporter must be
used with a BatchSpanProcessor.

.. code:: python

from opentelemetry import trace
from opentelemetry.sdk.trace.export import BatchSpanProcessor
from opentelemetry.exporter.richconsole import RichConsoleExporter
from opentelemetry.sdk.trace import TracerProvider

trace.set_tracer_provider(TracerProvider())
tracer = trace.get_tracer(__name__)

tracer.add_span_processor(BatchSpanProcessor(RichConsoleExporter()))


API
---
.. _Rich: https://rich.readthedocs.io/
.. _OpenTelemetry: https://github.com/open-telemetry/opentelemetry-python/
"""
# pylint: disable=import-error

import datetime
import typing
from typing import Optional

import opentelemetry.trace
from opentelemetry.sdk.trace import ReadableSpan
from opentelemetry.sdk.trace.export import SpanExporter, SpanExportResult
from opentelemetry.semconv.trace import SpanAttributes
from rich.console import Console
from rich.syntax import Syntax
from rich.text import Text
from rich.tree import Tree


def ns_to_time(nanoseconds):
ts = datetime.datetime.utcfromtimestamp(nanoseconds / 1e9)
return ts.strftime("%H:%M:%S.%f")


class RichConsoleExporter(SpanExporter):
tonybaloney marked this conversation as resolved.
Show resolved Hide resolved
"""Implementation of :class:`SpanExporter` that prints spans to the
console.

Must be used within a BatchSpanProcessor
srikanthccv marked this conversation as resolved.
Show resolved Hide resolved
"""

def __init__(
self,
service_name: Optional[str] = None,
):
self.service_name = service_name
self.console = Console()

def _child_to_tree(self, child: Tree, span: ReadableSpan):
child.add(Text.from_markup(f"[bold cyan]Kind :[/bold cyan] {span.kind.name}"))
if not span.status.is_unset:
if not span.status.is_ok:
child.add(
Text.from_markup(
f"[bold cyan]Status :[/bold cyan] [red]{span.status.status_code}[/red]"
)
)
else:
child.add(
Text.from_markup(
f"[bold cyan]Status :[/bold cyan] {span.status.status_code}"
)
)
if span.status.description:
child.add(
Text.from_markup(
f"[bold cyan]Description :[/bold cyan] {span.status.description}"
)
)

if span.events:
events = child.add(
label=Text.from_markup(f"[bold cyan]Events :[/bold cyan] ")
)
for event in span.events:
event_node = events.add(Text(event.name))
for k, v in event.attributes.items():
event_node.add(
Text.from_markup(f"[bold cyan]{k} :[/bold cyan] {v}")
)
if span.attributes:
attributes = child.add(
label=Text.from_markup(f"[bold cyan]Attributes :[/bold cyan] ")
)
for attribute in span.attributes:
if attribute == SpanAttributes.DB_STATEMENT:
attributes.add(
Text.from_markup(f"[bold cyan]{attribute} :[/bold cyan] ")
)
attributes.add(Syntax(span.attributes[attribute], "sql"))
else:
attributes.add(
Text.from_markup(
f"[bold cyan]{attribute} :[/bold cyan] {span.attributes[attribute]}"
)
)

def export(self, spans: typing.Sequence[ReadableSpan]) -> SpanExportResult:
if not spans:
return SpanExportResult.SUCCESS
tree = Tree(
label=f"Trace {opentelemetry.trace.format_trace_id(spans[0].context.trace_id)}"
)
parents = {}
root_spans = [span for span in spans if not span.parent]
for span in root_spans:
tonybaloney marked this conversation as resolved.
Show resolved Hide resolved
child = tree.add(
label=Text.from_markup(
f"[blue][{ns_to_time(span.start_time)}][/blue] [bold]{span.name}[/bold], span {opentelemetry.trace.format_span_id(span.context.span_id)}"
)
)
parents[span.context.span_id] = child
self._child_to_tree(child, span)

child_spans = [span for span in spans if span.parent]
for span in child_spans:
if span.parent.span_id not in parents:
tonybaloney marked this conversation as resolved.
Show resolved Hide resolved
child = tree.add(
label=Text.from_markup(
f"[blue][{ns_to_time(span.start_time)}][/blue] [bold]{span.name}[/bold], span {opentelemetry.trace.format_span_id(span.context.span_id)}"
)
)
else:
child = parents[span.parent.span_id].add(
label=Text.from_markup(
f"[blue][{ns_to_time(span.start_time)}][/blue] [bold]{span.name}[/bold], span {opentelemetry.trace.format_span_id(span.context.span_id)}"
)
)
parents[span.context.span_id] = child
self._child_to_tree(child, span)

self.console.print(tree)
return SpanExportResult.SUCCESS
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
# Copyright The OpenTelemetry Authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

__version__ = "0.24b0"