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

feat(stats): Arbitrum charts #1154

Merged
merged 17 commits into from
Dec 18, 2024
1 change: 1 addition & 0 deletions stats/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,7 @@ by enabling word wrapping
| `STATS__CONDITIONAL_​START__INTERNAL_​TRANSACTIONS_RATIO__​THRESHOLD` | | Value for `internal_​transactions_​ratio` threshold | `0.98` |
| `STATS__IGNORE_​BLOCKSCOUT_API_ABSENCE` | | Disable requirement for blockscout api url setting. Turns off corresponding features if the api setting is not set | `false` |
| `STATS__DISABLE_​INTERNAL_TRANSACTIONS` | | Disable functionality that utilizes internal transactions. In particular, disable internal transactions ratio check for starting the service and related charts (`newContracts`, `lastNewContracts`, and `contractsGrowth`). It has a higher priority than config files and respective envs. | `false` |
| `STATS__ENABLE_​ALL_ARBITRUM` | | Enable Arbitrum-specific charts. Variable for convenience only, the same can be done manually in configs. | `false` |

[anchor]: <> (anchors.envs.end.service)

Expand Down
15 changes: 15 additions & 0 deletions stats/config/charts.json
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,11 @@
"title": "Total txns",
"description": "All transactions including pending, dropped, replaced, failed transactions"
},
"total_operational_txns": {
"enabled": false,
"title": "Total operational txns",
"description": "'Total txns' without block creation transactions"
},
"yesterday_txns": {
"title": "Yesterday transactions",
"description": "Number of transactions yesterday (0:00 - 23:59 UTC)"
Expand Down Expand Up @@ -110,6 +115,16 @@
"title": "Number of transactions",
"description": "Cumulative transaction growth over time"
},
"new_operational_txns": {
"enabled": false,
"title": "New operational transactions",
"description": "Number of new transactions without block creation"
},
"operational_txns_growth": {
"enabled": false,
"title": "Number of operational transactions",
"description": "Cumulative transactions growth without block creation"
},
"txns_success_rate": {
"title": "Transaction success rate",
"description": "Success rate for all included transactions"
Expand Down
3 changes: 3 additions & 0 deletions stats/config/layout.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
"total_native_coin_transfers",
"total_tokens",
"total_txns",
"total_operational_txns",
"total_verified_contracts",
"yesterday_txns"
],
Expand All @@ -36,6 +37,8 @@
"new_txns",
"txns_fee",
"txns_growth",
"new_operational_txns",
"operational_txns_growth",
"txns_success_rate"
]
},
Expand Down
1 change: 1 addition & 0 deletions stats/config/update_groups.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
"total_blocks_group": "0 0 */2 * * * *",
"total_tokens_group": "0 0 18 * * * *",
"total_txns_group": "0 5 */2 * * * *",
"total_operational_txns_group": "0 5 1 * * * *",
"yesterday_txns_group": "0 8 0 * * * *",
"active_recurring_accounts_daily_recurrence_60_days_group": "0 0 2 * * * *",
"active_recurring_accounts_daily_recurrence_90_days_group": "0 20 2 * * * *",
Expand Down
262 changes: 262 additions & 0 deletions stats/config/utils/find_free_timeslot.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,262 @@
import tkinter as tk
bragov4ik marked this conversation as resolved.
Show resolved Hide resolved
from tkinter import ttk, filedialog, messagebox
import json
from datetime import datetime, timedelta
from croniter import croniter
import colorsys
import os
from tkcalendar import Calendar
from typing import Dict, List, Tuple

class CronVisualizerGUI:
def __init__(self, root):
self.root = root
self.root.title("Cron Schedule Visualizer")
self.root.geometry("1200x800")

self.schedules = {}
self.canvas_width = 1000
self.canvas_height = 200
self.hour_width = self.canvas_width // 24
self.selected_date = datetime.now()
self.default_duration = 20 # Duration in minutes

# Add default path
default_path = "../update_groups.json"
if os.path.exists(default_path):
try:
with open(default_path, 'r') as f:
data = json.load(f)
self.schedules = data.get('schedules', {})
except Exception as e:
print(f"Failed to load default file: {str(e)}")

self.setup_gui()

if self.schedules:
self.update_visualization()
self.update_schedule_list()

def setup_gui(self):
# Top frame for file selection and controls
top_frame = ttk.Frame(self.root, padding="10")
top_frame.pack(fill=tk.X)

ttk.Button(top_frame, text="Load JSON File", command=self.load_json).pack(side=tk.LEFT, padx=5)

self.ignore_days_var = tk.BooleanVar()
ttk.Checkbutton(top_frame, text="Ignore day parameters",
variable=self.ignore_days_var,
command=self.update_visualization).pack(side=tk.LEFT, padx=5)

# Duration control
ttk.Label(top_frame, text="Duration (minutes):").pack(side=tk.LEFT, padx=5)
self.duration_var = tk.StringVar(value=str(self.default_duration))
duration_entry = ttk.Entry(top_frame, textvariable=self.duration_var, width=5)
duration_entry.pack(side=tk.LEFT, padx=5)
duration_entry.bind('<Return>', lambda e: self.update_visualization())
ttk.Button(top_frame, text="Update", command=self.update_visualization).pack(side=tk.LEFT, padx=5)

# Calendar widget
calendar_frame = ttk.Frame(self.root, padding="10")
calendar_frame.pack(fill=tk.X)

self.calendar = Calendar(calendar_frame, selectmode='day',
year=self.selected_date.year,
month=self.selected_date.month,
day=self.selected_date.day)
self.calendar.pack(side=tk.LEFT)
self.calendar.bind('<<CalendarSelected>>', self.on_date_select)

# Timeline canvas
canvas_frame = ttk.Frame(self.root, padding="10")
canvas_frame.pack(fill=tk.BOTH, expand=True)

self.canvas = tk.Canvas(canvas_frame,
width=self.canvas_width,
height=self.canvas_height,
bg='white')
self.canvas.pack(fill=tk.BOTH, expand=True)

# Bind mouse motion for hover effect
self.canvas.bind('<Motion>', self.on_hover)

# Schedule list
list_frame = ttk.Frame(self.root, padding="10")
list_frame.pack(fill=tk.BOTH, expand=True)

self.schedule_list = ttk.Treeview(list_frame, columns=('Schedule', 'Times'),
show='headings')
self.schedule_list.heading('Schedule', text='Schedule Name')
self.schedule_list.heading('Times', text='Execution Times')
self.schedule_list.pack(fill=tk.BOTH, expand=True)

# Status bar
self.status_var = tk.StringVar()
status_bar = ttk.Label(self.root, textvariable=self.status_var)
status_bar.pack(fill=tk.X, pady=5)

def convert_7field_to_5field(self, cron_str: str) -> str:
"""Convert 7-field cron (with seconds and years) to 5-field format."""
fields = cron_str.split()
if len(fields) == 7:
return ' '.join(fields[1:-1])
return cron_str

def load_json(self):
file_path = filedialog.askopenfilename(
filetypes=[("JSON files", "*.json"), ("All files", "*.*")])
if not file_path:
return

try:
with open(file_path, 'r') as f:
data = json.load(f)
self.schedules = data.get('schedules', {})
self.update_visualization()
self.update_schedule_list()
except Exception as e:
messagebox.showerror("Error", f"Failed to load file: {str(e)}")

def get_color(self, value: int, max_value: int) -> str:
"""Generate color based on value intensity."""
if max_value == 0:
return "#FFFFFF"

# Convert from HSV to RGB (using red hue, varying saturation)
hue = 0 # Red
saturation = min(value / max_value, 1.0)
value = 1.0 # Brightness
rgb = colorsys.hsv_to_rgb(hue, saturation, value)

return f"#{int(rgb[0]*255):02x}{int(rgb[1]*255):02x}{int(rgb[2]*255):02x}"

def parse_cron_schedule(self, schedule: str, target_date: datetime) -> List[datetime]:
"""Parse cron schedule and return list of times it occurs in 24 hours."""
if self.ignore_days_var.get():
parts = schedule.split()
parts[3:] = ['*'] * len(parts[3:])
schedule = ' '.join(parts)

schedule = self.convert_7field_to_5field(schedule)
base = target_date.replace(hour=0, minute=0, second=0, microsecond=0)
next_day = base + timedelta(days=1)

try:
cron = croniter(schedule, base)
times = []
next_time = cron.get_next(datetime)

while next_time < next_day:
times.append(next_time)
next_time = cron.get_next(datetime)

return times
except ValueError:
return []

def get_task_overlaps(self) -> List[List[str]]:
"""Calculate overlapping tasks for each minute of the day."""
try:
duration = int(self.duration_var.get())
except ValueError:
duration = self.default_duration

# Initialize timeline with empty lists for each minute
timeline = [[] for _ in range(24 * 60)]

# For each schedule, add its task duration to the timeline
for name, schedule in self.schedules.items():
start_times = self.parse_cron_schedule(schedule, self.selected_date)

for start_time in start_times:
start_minute = start_time.hour * 60 + start_time.minute

# Add the task name to each minute it runs
for minute in range(start_minute, min(start_minute + duration, 24 * 60)):
timeline[minute].append(name)

return timeline

def update_visualization(self):
self.canvas.delete('all')

# Draw hour lines and labels
for hour in range(25):
x = hour * self.hour_width
self.canvas.create_line(x, 0, x, self.canvas_height, fill='gray')
if hour < 24:
self.canvas.create_text(x + self.hour_width/2, self.canvas_height - 20,
text=f"{hour:02d}:00")

# Get timeline with overlaps
timeline = self.get_task_overlaps()
max_overlaps = max(len(tasks) for tasks in timeline)

# Draw visualization
for minute in range(24 * 60):
hour = minute // 60
minute_in_hour = minute % 60

x = hour * self.hour_width + (minute_in_hour * self.hour_width / 60)
count = len(timeline[minute])

if count > 0:
color = self.get_color(count, max_overlaps)
x2 = x + self.hour_width / 60

self.canvas.create_rectangle(
x, 20,
x2, self.canvas_height - 40,
fill=color, outline='',
tags=('time_slot', f'minute_{minute}',
f'count_{count}',
f'tasks_{"/".join(timeline[minute])}') # Change separator to '/'
)

self.status_var.set(f"Maximum concurrent tasks: {max_overlaps}")

def update_schedule_list(self):
self.schedule_list.delete(*self.schedule_list.get_children())
for name, schedule in self.schedules.items():
times = self.parse_cron_schedule(schedule, self.selected_date)
if times or self.ignore_days_var.get():
time_str = ', '.join(t.strftime('%H:%M') for t in times)
self.schedule_list.insert('', 'end', values=(name, time_str))

def on_date_select(self, event=None):
date = self.calendar.get_date()
self.selected_date = datetime.strptime(date, '%m/%d/%y')
self.update_visualization()
self.update_schedule_list()

def on_hover(self, event):
x, y = event.x, event.y

if 20 <= y <= self.canvas_height - 40:
hour = int(x // self.hour_width)
minute_in_hour = int((x % self.hour_width) / (self.hour_width / 60))
minute_index = hour * 60 + minute_in_hour

if 0 <= minute_index < 24 * 60:
time_str = f"{hour:02d}:{minute_in_hour:02d}"
items = self.canvas.find_overlapping(x-1, 20, x+1, self.canvas_height-40)
if items:
for item in items:
tags = self.canvas.gettags(item)
# Fix 1: Check if we have a tasks tag before accessing index 3
tasks_tag = next((tag for tag in tags if tag.startswith('tasks_')), None)
if tasks_tag:
tasks = tasks_tag[6:].split('/') # Fix 2: Change separator to '/'
count = len(tasks)
task_list = ', '.join(tasks)
self.status_var.set(
f"Time: {time_str} - {count} concurrent tasks: {task_list}")
break
else:
self.status_var.set(f"Time: {time_str} - No tasks")

if __name__ == "__main__":
root = tk.Tk()
app = CronVisualizerGUI(root)
root.mainloop()
3 changes: 3 additions & 0 deletions stats/config/utils/requirements.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
tkinter>=3.13.1
tkcalendar>=1.6.1
croniter>=2.0.3
11 changes: 11 additions & 0 deletions stats/stats-server/src/runtime_setup.rs
Original file line number Diff line number Diff line change
Expand Up @@ -261,6 +261,7 @@ impl RuntimeSetup {
Arc::new(TotalBlocksGroup),
Arc::new(TotalTokensGroup),
Arc::new(TotalTxnsGroup),
Arc::new(TotalOperationalTxnsGroup),
Arc::new(YesterdayTxnsGroup),
Arc::new(ActiveRecurringAccountsDailyRecurrence60DaysGroup),
Arc::new(ActiveRecurringAccountsMonthlyRecurrence60DaysGroup),
Expand Down Expand Up @@ -351,6 +352,16 @@ impl RuntimeSetup {
("AverageGasPriceGroup", vec!["newTxns_DAY", "newTxns_MONTH"]),
("AverageTxnFeeGroup", vec!["newTxns_DAY", "newTxns_MONTH"]),
("TxnsSuccessRateGroup", vec!["newTxns_DAY", "newTxns_MONTH"]),
// total blocks and total txns have their own respective groups
(
"TotalOperationalTxnsGroup",
vec!["totalBlocks_DAY", "totalTxns_DAY"],
),
// the operational txns charts that depend on `newTxns_DAY` are
// rarely turned on, also `newTxns_DAY` is not that expensive to
// compute, therefore this solution is ok (to not introduce
// more update groups if not necessary)
("NewBlocksGroup", vec!["newTxns_DAY"]),
]
.map(|(group_name, allowed_missing)| {
(
Expand Down
3 changes: 2 additions & 1 deletion stats/stats-server/src/server.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use crate::{
health::HealthService,
read_service::ReadService,
runtime_setup::RuntimeSetup,
settings::{handle_disable_internal_transactions, Settings},
settings::{handle_disable_internal_transactions, handle_enable_all_arbitrum, Settings},
update_service::UpdateService,
};

Expand Down Expand Up @@ -66,6 +66,7 @@ pub async fn stats(mut settings: Settings) -> Result<(), anyhow::Error> {
let mut charts_config = read_charts_config(&settings.charts_config)?;
let layout_config = read_layout_config(&settings.layout_config)?;
let update_groups_config = read_update_groups_config(&settings.update_groups_config)?;
handle_enable_all_arbitrum(settings.enable_all_arbitrum, &mut charts_config);
handle_disable_internal_transactions(
settings.disable_internal_transactions,
&mut settings.conditional_start,
Expand Down
Loading
Loading