Add mix emission analysis (#55)

This commit is contained in:
Youngjoon Lee 2024-11-18 10:31:27 +09:00 committed by GitHub
parent 091a60ae4e
commit 319450ed59
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 88 additions and 21 deletions

56
scripts/emission.py Normal file
View File

@ -0,0 +1,56 @@
import argparse
import json
from collections.abc import Iterable
from typing import Any
import matplotlib
import matplotlib.pyplot as plt
import pandas as pd
import mixlog
def plot_emissions(input_stream: Iterable[str], plot_path: str) -> None:
df = pd.DataFrame(emission_records(input_stream))
plt.figure(figsize=(12, 6))
plt.scatter(df["step_id"], df["node_id"], c="red", marker="x", alpha=0.6)
plt.xlabel("Step ID")
plt.ylabel("Node ID")
plt.title("Distribution of Emissions")
plt.tight_layout()
plt.savefig(plot_path)
if matplotlib.is_interactive():
plt.show()
def emission_records(input_stream: Iterable[str]) -> list[Any]:
records = []
for line in input_stream:
try:
record = json.loads(line)
except json.JSONDecodeError:
continue
if "emission_type" in record:
records.append(record)
return records
if __name__ == "__main__":
parser = argparse.ArgumentParser(description="Mix emission analysis")
parser.add_argument(
"--log-path",
nargs="?",
type=str,
help="An input log file path. If not provided, input will be read from stdin.",
)
parser.add_argument(
"--plot-png-path", required=True, type=str, help="An output plot PNG file path"
)
args = parser.parse_args()
input = mixlog.get_input_stream(args.log_path)
plot_emissions(input, args.plot_png_path)

View File

@ -6,6 +6,8 @@ from typing import Dict, Optional
import statistics
import argparse
import mixlog
from json_stream.base import TransientStreamingJSONObject
JsonStream = Iterable[TransientStreamingJSONObject]
@ -83,26 +85,6 @@ def parse_record_stream(record_stream: Iterable[str]) -> MessageStorage:
return storage
def line_to_json_stream(record_stream: Iterable[str]) -> Iterable[str]:
for record in record_stream:
bracket_pos = record.rfind("{")
yield record[bracket_pos:]
def get_pipe_stream() -> Iterable[str]:
yield from sys.stdin
def get_file_stream(input_filename) -> Iterable[str]:
with open(input_filename, "r") as file:
yield from file
def get_input_stream(input_filename: Optional[str]) -> Iterable[str]:
stream = get_file_stream(input_filename) if input_filename is not None else get_pipe_stream()
return line_to_json_stream(stream)
def build_argument_parser() -> argparse.ArgumentParser:
parser = argparse.ArgumentParser(description="Log analysis for nomos-simulations.")
parser.add_argument(
@ -123,7 +105,7 @@ if __name__ == "__main__":
argument_parser = build_argument_parser()
arguments = argument_parser.parse_args()
input_stream = get_input_stream(arguments.input_file)
input_stream = mixlog.get_input_stream(arguments.input_file)
messages = parse_record_stream(input_stream)
results = compute_results(messages, arguments.step_duration)

27
scripts/mixlog.py Normal file
View File

@ -0,0 +1,27 @@
import sys
from collections.abc import Iterable
from typing import Optional
def line_to_json_stream(record_stream: Iterable[str]) -> Iterable[str]:
for record in record_stream:
bracket_pos = record.rfind("{")
yield record[bracket_pos:]
def get_pipe_stream() -> Iterable[str]:
yield from sys.stdin
def get_file_stream(input_filename) -> Iterable[str]:
with open(input_filename, "r") as file:
yield from file
def get_input_stream(input_filename: Optional[str]) -> Iterable[str]:
stream = (
get_file_stream(input_filename)
if input_filename is not None
else get_pipe_stream()
)
return line_to_json_stream(stream)

2
scripts/requirements.txt Normal file
View File

@ -0,0 +1,2 @@
matplotlib==3.9.2
pandas==2.2.3