Metadata-Version: 2.1
Name: trazer
Version: 0.2.0
Summary: A lightweight trace analysis framework.
Home-page: https://github.com/starsoi/trazer
Author: Hongyan Zhang
Author-email: starsoi@gmail.com
License: UNKNOWN
Platform: UNKNOWN
Classifier: Programming Language :: Python :: 3.8
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Topic :: Software Development :: Embedded Systems
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Intended Audience :: Developers
Requires-Python: >=3.8
Description-Content-Type: text/markdown
License-File: LICENSE

# Trazer

![Test](https://github.com/starsoi/trazer/actions/workflows/main.yml/badge.svg)
![Doctest](https://github.com/starsoi/trazer/actions/workflows/doctest.yml/badge.svg)
[![Coverage](https://codecov.io/gh/starsoi/trazer/branch/master/graph/badge.svg?token=HVX3PFO8RF)](https://codecov.io/gh/starsoi/trazer)
[![PyPI version](https://badge.fury.io/py/trazer.svg)](https://pypi.org/project/trazer)
[![Documentation Status](https://readthedocs.org/projects/trazer/badge/?version=latest)](https://trazer.readthedocs.io/en/latest/?badge=latest)

A lightweight trace analysis framework (**tra**ce analy**zer**) for execution and network traces,
focusing on event chain analysis.

## Getting Started

### Prerequisites

- Python >= 3.8

### Installation

```bash
pip install trazer
```

## Usage

### Create Trace and Add Events

```python
>>> from trazer import Trace, TraceEventDurationBegin, TraceEventDurationEnd
>>> trace = Trace()
>>> trace.add_event(TraceEventDurationBegin('my_event', 1.0, pid=0, tid=0))  # my_event begins at 1.0 s
>>> trace.add_event(TraceEventDurationEnd('my_event', 2.0, pid=0, tid=0))  # my_event ends at 2.0 s

```

### Export Trace to Chrome Tracing JSON

```python
>>> from io import StringIO
>>> s = StringIO()
>>> trace.to_tef_json(file_like=s)  # Exported timestamps are in milliseconds by default
>>> print(s.getvalue())
{
    "traceEvents": [
        {
            "name": "my_event",
            "ts": 1000.0,
            "pid": 0,
            "tid": 0,
            "args": {},
            "ph": "B"
        },
        {
            "name": "my_event",
            "ts": 2000.0,
            "pid": 0,
            "tid": 0,
            "args": {},
            "ph": "E"
        }
    ],
    "displayTimeUnit": "ms"
}

```

The exported `.json` file can be opened in the trace visualization tools `chrome://tracing` in Chrome
and [Perfetto](https://ui.perfetto.dev).

### Match Event Chains

Different related events in a trace can be merged and represented as an event chain at a higher hierarchical level.

An event chain is described using an event pattern, where the following symbols have special interpretation:

- `+` following an event name: the event begins.
- `-` following an event name: the event ends.
- `*`: arbitrary events, excluding repetitions.

```python
>>> from trazer import Trace, TraceEventDurationBegin, TraceEventDurationEnd
>>> trace = Trace()

# Add an event sequence: event1 begins, event2 begins, event2 ends, event1 ends
>>> trace.add_events([
... TraceEventDurationBegin('event1', 1),
... TraceEventDurationBegin('event2', 2),
... TraceEventDurationEnd('event2', 3),
... TraceEventDurationEnd('event1', 4)
... ])

>>> print(trace)
[1 ms]: event1 (B)
[2 ms]: event2 (B)
[3 ms]: event2 (E)
[4 ms]: event1 (E)

>>> from trazer import TraceAnalyzer
>>> trace_analyzer = TraceAnalyzer(trace)

# We want to find the event chains matching the sequence:
# event1 begins -> event2 begins -> event2 ends -> event1 ends
>>> trace_analyzer.match('event1+event2+event2-event1-', 'event_chain')
[[1 - 4 ms]: event_chain (4 events)]

# Or use an alternative pattern employing wildcards.
# We want to find the event chains that begins with event1 and ends with event1.
>>> trace_analyzer = TraceAnalyzer(trace)
>>> trace_analyzer.match('event1+*-event1-', 'event_chain')
[[1 - 4 ms]: event_chain (4 events)]

```

### Export Trace with Event Chains

The event chains can be visualized together with the original trace in the same view.
For visualization in `chrome://tracing` or [Perfetto](https://ui.perfetto.dev), a dedicated process ID needs to be assigned to
the event chains, such that they will be displayed separately from the original trace.

```python
trace_analyzer.to_tef_json(5555)  # Process ID = 5555
```

## Contributing

1. Install development dependencies

```bash
pip install -r requirements-dev.txt
```

2. Install runtime dependencies

```bash
pip install -r requirements.txt
```

3. Setup pre-commit hook (formatting with black and linting with flake8)

```bash
pre-commit install
```

4. Make sure all tests are passed by running `pytest`


