Metadata-Version: 2.4
Name: chronologix
Version: 0.2.1
Summary: Fully async, modular logging system for Python with time-based rollover, mirror streams, and log level filtering.
Author: Fractalwillie
License: MIT License
        
        Copyright (c) [2025] [Fractalwillie]
        
        Permission is hereby granted, free of charge, to any person obtaining a copy
        of this software and associated documentation files (the "Software"), to deal
        in the Software without restriction, including without limitation the rights
        to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
        copies of the Software, and to permit persons to whom the Software is
        furnished to do so, subject to the following conditions:
        
        The above copyright notice and this permission notice shall be included in all
        copies or substantial portions of the Software.
        
        THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
        IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
        FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
        AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
        LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
        OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
        SOFTWARE.
Project-URL: Homepage, https://github.com/fractalwillie/chronologix
Project-URL: Issues, https://github.com/fractalwillie/chronologix/issues
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Operating System :: OS Independent
Classifier: License :: OSI Approved :: MIT License
Requires-Python: >=3.7
Description-Content-Type: text/markdown
License-File: LICENSE
Dynamic: license-file

# Chronologix

Chronologix is a fully asynchronous, modular logging system for Python.

It writes structured log files across multiple named streams, supports time-based chunking, and avoids the standard logging module completely.

---

## Features

-  Fully async logging 
-  Time-based rollover (e.g. every 24h, 1h, 15min)
-  Log stream isolation (e.g. `errors`, `debug`, `events`)
-  Configurable mirror streams (`errors` → `debug`)
-  Optional per-stream log level filtering (`INFO`, `ERROR`, etc.)
-  Safe, stateless async file writes
-  Config validation with clear error feedback
-  Custom log paths via `str` or `pathlib.Path`
-  Predictable file and folder structure for automated processing

---

## Installation

Chronologix requires **Python 3.7+**.

```bash
pip install chronologix
```

---

## Usage example

```python
from chronologix import LogConfig, LogManager

config = LogConfig(
    base_log_dir="my_logs",                 # can also be a pathlib.Path
    interval="1h",                          # rollover interval
    log_streams=["app", "errors", "audit"], # named log streams
    mirror_map={"errors": ["app"]},         # errors are mirrored into "app"
    min_log_levels={"app": "INFO", "errors": "ERROR"}, # logs filtered by level
    timestamp_format="%H:%M:%S.%f"          # timestamp format
)

logger = LogManager(config)

async def divide(a, b):
    try:
        result = a / b
        await logger.log(f"Division result: {result}", target="app", level="INFO")
    except Exception as e:
        await logger.log(f"Exception occurred: {e}", target="errors", level="ERROR")

async def main():
    await logger.start()
    await logger.log("Starting batch job", target="app", level="INFO")
    await logger.log("Auditing step 1", target="audit") # called without "level" since "audit" isn't set in min_log_levels
    await divide(10, 0)  # this will raise and log to both "errors" and "app"
    await logger.stop()

```
This example will produce following:
- A new folder per hour like: 2025-05-04__14-00/
- Three log files inside: app.log, errors.log, audit.log
- The exception will be logged to both errors.log and app.log
- The log level filtering will be applied only to "app" and "errors" streams
- The audit message will only go to audit.log with no mirroring nor log level filtering

---

## Path structure

You can set the log output directory using either a string path or a `pathlib.Path` object.

Examples:
```python
LogConfig(base_log_dir="logs")  # relative to current working dir
LogConfig(base_log_dir="/var/log/chronologix")  # absolute path (Linux)
LogConfig(base_log_dir=Path("~/.chronologix").expanduser())  # user home dir
```
Chronologix will create any missing directories automatically.

---

## Intervals

The `interval` controls how frequently Chronologix creates a new folder and rotates the log files.

Supported values:
- `"24h"`
- `"12h"`
- `"6h"`
- `"3h"`
- `"1h"`
- `"30m"`
- `"15m"`
- `"5m"`

Each interval corresponds to a different granularity of time-based chunking.
- `interval="24h"` → folders like `2025-05-04/`
- `interval="1h"` → folders like `2025-05-04__14-00/`

---

## Log streams

Log streams define the named `.log` files Chronologix will manage.

Each stream corresponds to a separate log file inside each time-based folder.

Example:
```python
log_streams=["app", "errors", "audit"]
```
This would create:
```lua
my_logs/
└── 2025-05-04/
    ├── app.log
    ├── errors.log
    └── audit.log
```
Each call to .log(message, target=...) writes to the stream you specify.

You can define as many log streams as needed or just a single one.

Example:
```python
LogConfig(
    log_streams=["app"],
    mirror_map={}
)
```
This will create a single `app.log` file per interval.
Mirroring is optional, and is not required when using only one stream.

---

## Mirroring

Mirroring can be configured like this:
```python
mirror_map = {
    "errors": ["app"],    # messages logged to "errors" will mirror to "app"
    "debug":  ["all"]     # # messages logged to "debug" will mirror to "all"
}
```
Mirroring is optional. Any stream can exist without mirrors, and mirrors can point to multiple targets.

### Flat mirroring only
Mirroring is one-level deep by design.
A message logged to "trace" and mirrored to "debug" will not be re-mirrored from "debug" to another stream.
This avoids complex logic, recursive loops, and keeps the output clean.

---

## Log Levels

Chronologix supports configurable log level thresholds for each stream.
This allows you to filter out lower-priority messages from specific log files.

### Hierarchy
Levels are evaluated by their severity:
```python
LOG_LEVELS = {
    "TRACE": 5,    # most verbose
    "DEBUG": 10,
    "INFO": 20,
    "WARNING": 30,
    "ERROR": 40,
    "CRITICAL": 50 # most severe
}
```
Each stream only logs messages with severity greater than or equal to its configured threshold.

Example:
```python
log_streams=["stdout", "errors", "debug"],
min_log_levels={
    "stdout": "INFO",     # logs INFO, WARNING, ERROR, CRITICAL
    "errors": "ERROR",    # logs ERROR, CRITICAL
    "debug": "DEBUG",     # logs DEBUG, INFO, WARNING, ERROR, CRITICAL
}
```
If a message is below a stream's threshold, it will be skipped.

This also applies to mirrored logs.
A message will only be mirrored to a stream if that stream accepts its level.

### Optional usage
- Log levels are fully optional.
- If no level is passed to .log(...), the message will still be written to any stream that does not have a threshold.

Messages without levels are logged like:
```lua
[14:02:19] Something went wrong
```
While messages with levels include:
```lua
[14:02:19] [ERROR] Something went wrong

```

#### Why no `NOTSET`?
Unlike some logging systems, Chronologix does not include `NOTSET`.
I decided to go with `TRACE` as the lowest level instead.
It allows for the log() function to be invoked without level argument and ignore the log levels completely.

### Using Chronologix without log levels
If you don’t want log level filtering simply skip `min_log_levels` and use the default config.
Or only define thresholds for select streams.

Example:
```python
log_streams=["stdout", "audit"],
min_log_levels={"stdout": "INFO"}

await logger.log("Something happened", target="stdout", level="WARNING")
await logger.log("Just some note", target="audit")  # no level, still works
```

---

## Timestamp formatting

Customize timestamp formatting using any valid strftime directive.

Examples:

    - %H:%M:%S → 14:02:19

    - %H:%M:%S.%f → 14:02:19.123456

    - %Y-%m-%d %H:%M:%S → 2025-05-04 14:02:19

Invalid formats are rejected with a descriptive LogConfigError.

---

## Log structure

```lua
my_logs/
└── 2025-05-04__14-00/
    ├── app.log
    ├── errors.log
    └── audit.log
└── 2025-05-04__15-00/
    ├── app.log
    ├── errors.log
    └── audit.log
```
Folders are aligned to the start of the interval (__14-00) and created ahead of time to mitigate latency for smooth rollover.

---

## Default config

If you use the default constructor, Chronologix behaves like this:
```python
from chronologix import LogConfig

config = LogConfig()
logger = LogManager(config)
await logger.start()
```
Which is equivalent to:
```python
LogConfig(
    base_log_dir="logs",
    interval="24h",
    log_streams=["all", "errors"],
    mirror_map={"errors": ["all"]},
    min_log_levels={},
    timestamp_format="%H:%M:%S"
)
```

---

## But why?

The idea to build this package came from direct need while working on my private trading software. 
I hadn't found anything that would check all the boxes and satisfy my OCD, so I decided to build it myself. 
At first, it was just a module tailored for my program, but then I realized it could be useful for others. 
So it felt like the perfect opportunity to finally open source something.
The core of Chronologix is built on my original logging module, but I tried to make it as flexible as possible to cater to different needs.

---

## Contributing

Feel free to reach out if you have any suggestions or ideas. 
I'm open to collaboration and improvements.


