{"nodes":[{"code":"import os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nimport math\nimport random\nimport matplotlib.pyplot as plt\n\n# Device handling (required)\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data structure\nexperiment_data = {\n    \"BPI2012\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"BPI2017\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"ROAD\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n}\n\n# Data loading utilities (use provided helper)\nfrom ai_scientist.ideas.my_research_topic import load_datasets, pick_default_dataset\n\n\n# Reproducibility\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# Build prefixes\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    # Only complete transitions\n    df = df.copy()\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    # Build activity vocab\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}  # 0 for PAD\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n    # Case start time\n    case_groups = df.groupby(\"case_id\")\n    samples = []\n    for cid, g in case_groups:\n        g = g.sort_values(\"timestamp\")\n        times = g[\"timestamp\"].values\n        acts_ids = [act2id[a] for a in g[\"activity\"].astype(str).tolist()]\n        resources = g[\"resource\"].astype(str).tolist()\n        # per event features\n        # compute deltas in seconds\n        ts = pd.to_datetime(g[\"timestamp\"]).astype(\"int64\") // 10**9\n        deltas = np.diff(ts, prepend=ts[0])\n        since_start = ts - ts[0]\n        hours = pd.to_datetime(g[\"timestamp\"]).dt.hour.values / 23.0\n        weekdays = pd.to_datetime(g[\"timestamp\"]).dt.weekday.values / 6.0\n        working = (\n            (\n                (pd.to_datetime(g[\"timestamp\"]).dt.weekday < 5)\n                & (pd.to_datetime(g[\"timestamp\"]).dt.hour.between(8, 17))\n            ).astype(float)\n        ).values\n        # Prepare standardization (global later) but keep raw now\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        )  # shape [T,5]\n        T = len(acts_ids)\n        # generate prefixes up to T-1 (predict next)\n        for k in range(min_prefix_len, min(T, max_prefix_len) + 0):\n            # k is prefix length, predict act at index k\n            if k >= T:\n                break\n            if k < 1:\n                continue\n            seq_acts = acts_ids[:k]\n            seq_feats = feats[:k]\n            target = acts_ids[k] if k < T else None\n            if target is None:\n                continue\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": seq_acts,\n                    \"seq_feats\": seq_feats,\n                    \"target\": target,\n                    \"last_ts\": ts[k - 1],\n                    \"next_ts\": ts[k],\n                }\n            )\n    # Collect feature normalization stats over all feats\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    # Standardize first two columns (seconds) and keep the rest as already normalized [0,1]\n    dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n    ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n    for s in samples:\n        s[\"seq_feats\"] = s[\"seq_feats\"].copy()\n        if s[\"seq_feats\"].shape[0] > 0:\n            s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n            s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n    return samples, act2id, id2act, pad_id\n\n\n# Time-based split by case start time\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.zeros((pad_len, self.num_cont), dtype=np.float32)\n        feats_pad = np.vstack([feats_pad, feats.astype(np.float32)])\n        attn_mask = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad, dtype=torch.long),\n            \"feats\": torch.tensor(feats_pad, dtype=torch.float32),\n            \"mask\": torch.tensor(attn_mask, dtype=torch.float32),\n            \"y\": torch.tensor(s[\"target\"], dtype=torch.long),\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(\n        self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, num_layers=1, pad_idx=0\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim,\n            hidden_size=hidden,\n            batch_first=True,\n            num_layers=num_layers,\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(\n            hidden, vocab_size + 1\n        )  # includes PAD, but we won't predict PAD in metrics\n        self.pad_idx = pad_idx\n\n    def forward(self, acts, feats, mask):\n        # acts: [B,T], feats: [B,T,C]\n        x = self.emb(acts)  # [B,T,emb]\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h_last = h[-1]  # [B,H]\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)  # [B,V]\n        return logits\n\n\ndef collate_fn(batch):\n    # Already fixed-size in __getitem__\n    keys = batch[0].keys()\n    out = {k: torch.stack([b[k] for b in batch], dim=0) for k in keys}\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys, preds_top1, preds_probs = [], [], []\n    top3_correct = 0\n    n_total = 0\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            topk_vals, topk_idx = torch.topk(probs, k=min(3, probs.size(1)), dim=1)\n            # Exclude pad predictions in metrics by allowing as is; labels are always real activities\n            ys.extend(batch[\"y\"].detach().cpu().tolist())\n            preds_top1.extend(top1.detach().cpu().tolist())\n            preds_probs.append(probs.detach().cpu().numpy())\n            # top-3 correctness\n            for i in range(batch[\"y\"].size(0)):\n                if batch[\"y\"][i].item() in topk_idx[i].detach().cpu().tolist():\n                    top3_correct += 1\n            n_total += batch[\"y\"].size(0)\n    avg_loss = total_loss / max(1, n_total)\n    y_true = np.array(ys)\n    y_pred = np.array(preds_top1)\n    # Remove any PAD labels if present (shouldn't be)\n    mask = y_true != pad_idx\n    y_true = y_true[mask]\n    y_pred = y_pred[mask]\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    # Macro-F1 across seen classes in y_true\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except Exception:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n_total))\n    probs_concat = (\n        np.concatenate(preds_probs, axis=0)\n        if len(preds_probs) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return avg_loss, acc, f1, top3, y_true, y_pred, probs_concat\n\n\ndef train_one_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    # Time-based split\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    # Build samples for each split using only respective cases to prevent leakage in stats:\n    # We compute normalization using train only, so build on train then transform others with train stats. For simplicity we will build on full but re-standardize using train stats by rebuilding stats on train samples.\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    # Filter per split\n    samples_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    samples_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    samples_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # Recompute normalization using train samples only\n    if len(samples_train) > 0:\n        all_feats = np.concatenate(\n            [s[\"seq_feats\"] for s in samples_train if len(s[\"seq_feats\"]) > 0], axis=0\n        )\n        dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n        ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n\n        def norm_samples(samples):\n            for s in samples:\n                if s[\"seq_feats\"].shape[0] > 0:\n                    s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                    s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n        norm_samples(samples_train)\n        norm_samples(samples_val)\n        norm_samples(samples_test)\n    print(\n        f\"Samples train/val/test: {len(samples_train)}/{len(samples_val)}/{len(samples_test)}; vocab={len(act2id)}\"\n    )\n    if len(samples_train) == 0 or len(act2id) < 2:\n        print(\"Not enough data to train. Skipping.\")\n        return\n    ds_train = PrefixDataset(\n        samples_train, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_val = PrefixDataset(\n        samples_val, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_test = PrefixDataset(\n        samples_test, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    dl_train = DataLoader(\n        ds_train,\n        batch_size=batch_size,\n        shuffle=True,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_val = DataLoader(\n        ds_val,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_test = DataLoader(\n        ds_test,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    # Model\n    model = LSTMBaseline(\n        vocab_size=len(act2id),\n        emb_dim=64,\n        cont_dim=5,\n        hidden=128,\n        num_layers=1,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss().to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n    # Training loop\n    best_val_top3 = -1.0\n    best_state = None\n    hist = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for epoch in range(1, max_epochs + 1):\n        model.train()\n        total = 0\n        running_loss = 0.0\n        for batch in dl_train:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            running_loss += loss.item() * logits.size(0)\n            total += logits.size(0)\n        train_loss = running_loss / max(1, total)\n        val_loss, val_acc, val_f1, val_top3, y_true_v, y_pred_v, _ = evaluate(\n            model, dl_val, criterion, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        hist[\"train_loss\"].append(train_loss)\n        hist[\"val_loss\"].append(val_loss)\n        hist[\"val_top3\"].append(val_top3)\n        # Save to experiment_data\n        experiment_data[name][\"losses\"][\"train\"].append((epoch, train_loss))\n        experiment_data[name][\"losses\"][\"val\"].append((epoch, val_loss))\n        experiment_data[name][\"metrics\"][\"val\"].append(\n            (epoch, {\"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3})\n        )\n        experiment_data[name][\"epochs\"].append(epoch)\n        if val_top3 > best_val_top3:\n            best_val_top3 = val_top3\n            best_state = {k: v.cpu().clone() for k, v in model.state_dict().items()}\n    # Load best\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n    # Final eval on train/val/test\n    train_loss, train_acc, train_f1, train_top3, _, _, _ = evaluate(\n        model, dl_train, criterion, device, len(act2id), pad_id\n    )\n    val_loss, val_acc, val_f1, val_top3, y_true_v, y_pred_v, _ = evaluate(\n        model, dl_val, criterion, device, len(act2id), pad_id\n    )\n    test_loss, test_acc, test_f1, test_top3, y_true_t, y_pred_t, probs_t = evaluate(\n        model, dl_test, criterion, device, len(act2id), pad_id\n    )\n    print(\n        f\"[{name}] Train: loss={train_loss:.4f} acc={train_acc:.4f} f1={train_f1:.4f} top3={train_top3:.4f}\"\n    )\n    print(\n        f\"[{name}] Test:  loss={test_loss:.4f} acc={test_acc:.4f} f1={test_f1:.4f} top3={test_top3:.4f}\"\n    )\n    # Save metrics\n    experiment_data[name][\"metrics\"][\"train\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": train_loss,\n                \"acc\": train_acc,\n                \"macro_f1\": train_f1,\n                \"top3\": train_top3,\n            },\n        )\n    )\n    experiment_data[name][\"metrics\"][\"val\"].append(\n        (\n            \"final\",\n            {\"loss\": val_loss, \"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3},\n        )\n    )\n    experiment_data[name][\"metrics\"][\"test\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": test_loss,\n                \"acc\": test_acc,\n                \"macro_f1\": test_f1,\n                \"top3\": test_top3,\n            },\n        )\n    )\n    experiment_data[name][\"predictions\"] = y_pred_t.tolist()\n    experiment_data[name][\"ground_truth\"] = y_true_t.tolist()\n    # Plots\n    try:\n        plt.figure()\n        plt.plot(hist[\"train_loss\"], label=\"train_loss\")\n        plt.plot(hist[\"val_loss\"], label=\"val_loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"loss_curves_{name}.png\"))\n        plt.close()\n        plt.figure()\n        plt.plot(hist[\"val_top3\"], label=\"val_top3\")\n        plt.legend()\n        plt.title(f\"Val Top-3 Acc - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Acc\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"val_top3_{name}.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n    # Save confusion matrix-like data (optional)\n    try:\n        from sklearn.metrics import confusion_matrix\n\n        # Map labels to range of seen labels\n        cm = confusion_matrix(y_true_t, y_pred_t)\n        np.save(os.path.join(working_dir, f\"cm_{name}.npy\"), cm)\n    except Exception as e:\n        print(f\"[warn] Confusion matrix failed: {e}\")\n\n\ndef main():\n    datasets = load_datasets()\n    # Loop through loaded datasets but limit to first available for speed; still attempt all for completeness if small\n    for key, df in datasets.items():\n        # Clip to a manageable size if huge for runtime\n        # Keep time-order by cases\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep_cases = set(starts.iloc[:5000][\"case_id\"])\n                df_small = df[df[\"case_id\"].isin(keep_cases)].copy()\n            else:\n                df_small = df\n        except Exception:\n            df_small = df\n        train_one_dataset(\n            key, df_small, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Also compressed\n    np.savez_compressed(\n        os.path.join(working_dir, \"experiment_data_compressed.npz\"),\n        data=experiment_data,\n    )\n\n\n# Execute immediately\nmain()\n","plan":"We implement a basic predictive process monitoring pipeline for next-activity prediction using local XES logs via pm4py and the provided loader helper. We use a time-based split by sorting cases on their first event timestamp into train/val/test (70/15/15) to avoid label leakage. From each case, we build prefix-target pairs up to a maximum prefix length (e.g., 10), where the input is a sequence of activities plus simple time-aware features (delta time, time since start, hour, weekday, working time flag). Activities are embedded and concatenated with normalized continuous features per step, then fed to a 1-layer LSTM; the final hidden state feeds a linear classifier over activities. We report top-1 accuracy, macro-F1, and top-3 accuracy, and track validation loss each epoch. We save metrics, losses, predictions, and ground truth to working/experiment_data.npy and produce simple training curves. The implementation focuses on robustness and a minimal functional baseline; hyperparameters are conservative for runtime. If multiple datasets are available, we pick one by preference (BPI2017, then BPI2012, then ROAD), but will loop all loaded datasets for completeness.","overall_plan":"","plot_code":null,"plot_plan":null,"step":0,"id":"637162c2bceb4502a8e96d90218586d2","ctime":1757756008.4053643,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available in /workspace/data: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<24:45,  8.81it/s]","\rparsing log, completed traces ::   1%|1         | 145/13087 [00:00<00:16, 798.70it/s]","\rparsing log, completed traces ::   2%|2         | 286/13087 [00:00<00:12, 1058.76it/s]","\rparsing log, completed traces ::   3%|3         | 432/13087 [00:00<00:10, 1212.44it/s]","\rparsing log, completed traces ::   4%|4         | 574/13087 [00:00<00:09, 1285.85it/s]","\rparsing log, completed traces ::   6%|5         | 749/13087 [00:00<00:08, 1441.03it/s]","\rparsing log, completed traces ::   7%|6         | 895/13087 [00:00<00:08, 1435.73it/s]","\rparsing log, completed traces ::   8%|7         | 1041/13087 [00:00<00:08, 1443.19it/s]","\rparsing log, completed traces ::   9%|9         | 1199/13087 [00:00<00:08, 1479.06it/s]","\rparsing log, completed traces ::  10%|#         | 1348/13087 [00:01<00:07, 1481.32it/s]","\rparsing log, completed traces ::  12%|#1        | 1527/13087 [00:01<00:07, 1573.73it/s]","\rparsing log, completed traces ::  13%|#2        | 1685/13087 [00:01<00:07, 1549.92it/s]","\rparsing log, completed traces ::  14%|#4        | 1841/13087 [00:01<00:07, 1549.55it/s]","\rparsing log, completed traces ::  15%|#5        | 1997/13087 [00:01<00:12, 865.75it/s] ","\rparsing log, completed traces ::  16%|#6        | 2140/13087 [00:01<00:11, 973.73it/s]","\rparsing log, completed traces ::  17%|#7        | 2290/13087 [00:01<00:09, 1086.73it/s]","\rparsing log, completed traces ::  19%|#8        | 2464/13087 [00:01<00:08, 1240.23it/s]","\rparsing log, completed traces ::  20%|#9        | 2615/13087 [00:02<00:08, 1306.94it/s]","\rparsing log, completed traces ::  21%|##1       | 2763/13087 [00:02<00:07, 1324.78it/s]","\rparsing log, completed traces ::  22%|##2       | 2908/13087 [00:02<00:07, 1343.78it/s]","\rparsing log, completed traces ::  23%|##3       | 3053/13087 [00:02<00:07, 1366.05it/s]","\rparsing log, completed traces ::  24%|##4       | 3196/13087 [00:02<00:07, 1379.99it/s]","\rparsing log, completed traces ::  26%|##6       | 3432/13087 [00:02<00:05, 1659.94it/s]","\rparsing log, completed traces ::  28%|##7       | 3603/13087 [00:02<00:05, 1669.12it/s]","\rparsing log, completed traces ::  29%|##8       | 3773/13087 [00:02<00:05, 1643.24it/s]","\rparsing log, completed traces ::  30%|###       | 3940/13087 [00:02<00:05, 1622.12it/s]","\rparsing log, completed traces ::  32%|###1      | 4158/13087 [00:03<00:05, 1778.53it/s]","\rparsing log, completed traces ::  33%|###3      | 4338/13087 [00:03<00:08, 1000.66it/s]","\rparsing log, completed traces ::  35%|###4      | 4534/13087 [00:03<00:07, 1180.28it/s]","\rparsing log, completed traces ::  36%|###6      | 4713/13087 [00:03<00:06, 1306.16it/s]","\rparsing log, completed traces ::  37%|###7      | 4876/13087 [00:03<00:06, 1366.24it/s]","\rparsing log, completed traces ::  39%|###8      | 5043/13087 [00:03<00:05, 1437.17it/s]","\rparsing log, completed traces ::  40%|###9      | 5205/13087 [00:03<00:05, 1474.64it/s]","\rparsing log, completed traces ::  41%|####1     | 5366/13087 [00:04<00:05, 1480.65it/s]","\rparsing log, completed traces ::  42%|####2     | 5524/13087 [00:04<00:05, 1446.10it/s]","\rparsing log, completed traces ::  43%|####3     | 5683/13087 [00:04<00:04, 1483.65it/s]","\rparsing log, completed traces ::  45%|####4     | 5837/13087 [00:04<00:04, 1480.61it/s]","\rparsing log, completed traces ::  46%|####5     | 6012/13087 [00:04<00:04, 1556.42it/s]","\rparsing log, completed traces ::  47%|####7     | 6171/13087 [00:04<00:04, 1532.27it/s]","\rparsing log, completed traces ::  49%|####8     | 6349/13087 [00:04<00:04, 1599.11it/s]","\rparsing log, completed traces ::  50%|####9     | 6516/13087 [00:04<00:04, 1612.44it/s]","\rparsing log, completed traces ::  51%|#####1    | 6679/13087 [00:04<00:04, 1564.02it/s]","\rparsing log, completed traces ::  52%|#####2    | 6837/13087 [00:04<00:03, 1564.71it/s]","\rparsing log, completed traces ::  53%|#####3    | 6995/13087 [00:05<00:07, 770.95it/s] ","\rparsing log, completed traces ::  55%|#####4    | 7151/13087 [00:05<00:06, 904.26it/s]","\rparsing log, completed traces ::  56%|#####5    | 7299/13087 [00:05<00:05, 1016.37it/s]","\rparsing log, completed traces ::  57%|#####6    | 7436/13087 [00:05<00:05, 1081.43it/s]","\rparsing log, completed traces ::  58%|#####7    | 7571/13087 [00:05<00:04, 1139.53it/s]","\rparsing log, completed traces ::  59%|#####8    | 7710/13087 [00:05<00:04, 1198.71it/s]","\rparsing log, completed traces ::  60%|######    | 7863/13087 [00:06<00:04, 1285.85it/s]","\rparsing log, completed traces ::  61%|######1   | 8004/13087 [00:06<00:03, 1293.09it/s]","\rparsing log, completed traces ::  62%|######2   | 8156/13087 [00:06<00:03, 1353.39it/s]","\rparsing log, completed traces ::  64%|######3   | 8321/13087 [00:06<00:03, 1433.44it/s]","\rparsing log, completed traces ::  65%|######4   | 8473/13087 [00:06<00:03, 1456.26it/s]","\rparsing log, completed traces ::  66%|######5   | 8630/13087 [00:06<00:03, 1485.53it/s]","\rparsing log, completed traces ::  67%|######7   | 8784/13087 [00:06<00:02, 1491.68it/s]","\rparsing log, completed traces ::  68%|######8   | 8935/13087 [00:06<00:02, 1488.67it/s]","\rparsing log, completed traces ::  69%|######9   | 9086/13087 [00:06<00:02, 1463.15it/s]","\rparsing log, completed traces ::  71%|#######   | 9247/13087 [00:06<00:02, 1505.18it/s]","\rparsing log, completed traces ::  72%|#######1  | 9404/13087 [00:07<00:02, 1524.21it/s]","\rparsing log, completed traces ::  73%|#######3  | 9557/13087 [00:07<00:02, 1409.27it/s]","\rparsing log, completed traces ::  74%|#######4  | 9709/13087 [00:07<00:02, 1439.62it/s]","\rparsing log, completed traces ::  75%|#######5  | 9875/13087 [00:07<00:04, 760.54it/s] ","\rparsing log, completed traces ::  76%|#######6  | 10006/13087 [00:07<00:03, 853.96it/s]","\rparsing log, completed traces ::  78%|#######7  | 10155/13087 [00:07<00:02, 978.12it/s]","\rparsing log, completed traces ::  79%|#######8  | 10290/13087 [00:08<00:02, 1056.92it/s]","\rparsing log, completed traces ::  80%|#######9  | 10430/13087 [00:08<00:02, 1137.82it/s]","\rparsing log, completed traces ::  81%|########  | 10563/13087 [00:08<00:02, 1172.22it/s]","\rparsing log, completed traces ::  82%|########1 | 10714/13087 [00:08<00:01, 1257.40it/s]","\rparsing log, completed traces ::  83%|########3 | 10885/13087 [00:08<00:01, 1380.34it/s]","\rparsing log, completed traces ::  84%|########4 | 11054/13087 [00:08<00:01, 1463.77it/s]","\rparsing log, completed traces ::  86%|########5 | 11207/13087 [00:08<00:01, 1475.63it/s]","\rparsing log, completed traces ::  87%|########6 | 11381/13087 [00:08<00:01, 1551.64it/s]","\rparsing log, completed traces ::  88%|########8 | 11548/13087 [00:08<00:00, 1584.96it/s]","\rparsing log, completed traces ::  90%|########9 | 11724/13087 [00:08<00:00, 1634.26it/s]","\rparsing log, completed traces ::  91%|#########1| 11912/13087 [00:09<00:00, 1706.32it/s]","\rparsing log, completed traces ::  92%|#########2| 12085/13087 [00:09<00:00, 1693.93it/s]","\rparsing log, completed traces ::  94%|#########3| 12279/13087 [00:09<00:00, 1764.74it/s]","\rparsing log, completed traces ::  95%|#########5| 12457/13087 [00:09<00:00, 1751.83it/s]","\rparsing log, completed traces ::  97%|#########6| 12638/13087 [00:09<00:00, 1767.35it/s]","\rparsing log, completed traces ::  98%|#########7| 12816/13087 [00:09<00:00, 1750.81it/s]","\rparsing log, completed traces ::  99%|#########9| 12992/13087 [00:09<00:00, 1750.81it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:09<00:00, 1352.12it/s]","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<5:17:15,  1.66it/s]","\rparsing log, completed traces ::   0%|          | 62/31509 [00:00<04:26, 117.98it/s]","\rparsing log, completed traces ::   0%|          | 111/31509 [00:00<02:38, 197.62it/s]","\rparsing log, completed traces ::   0%|          | 153/31509 [00:01<03:07, 166.86it/s]","\rparsing log, completed traces ::   1%|          | 212/31509 [00:01<02:08, 243.78it/s]","\rparsing log, completed traces ::   1%|          | 267/31509 [00:01<01:42, 306.05it/s]","\rparsing log, completed traces ::   1%|1         | 324/31509 [00:01<01:25, 365.34it/s]","\rparsing log, completed traces ::   1%|1         | 379/31509 [00:01<01:15, 410.08it/s]","\rparsing log, completed traces ::   1%|1         | 435/31509 [00:01<01:09, 448.44it/s]","\rparsing log, completed traces ::   2%|1         | 489/31509 [00:01<01:05, 472.83it/s]","\rparsing log, completed traces ::   2%|1         | 542/31509 [00:01<01:03, 484.26it/s]","\rparsing log, completed traces ::   2%|1         | 599/31509 [00:01<01:00, 506.85it/s]","\rparsing log, completed traces ::   2%|2         | 661/31509 [00:02<00:57, 538.06it/s]","\rparsing log, completed traces ::   2%|2         | 723/31509 [00:02<00:54, 560.27it/s]","\rparsing log, completed traces ::   2%|2         | 781/31509 [00:02<00:55, 551.18it/s]","\rparsing log, completed traces ::   3%|2         | 838/31509 [00:02<00:55, 556.41it/s]","\rparsing log, completed traces ::   3%|2         | 895/31509 [00:02<00:54, 560.18it/s]","\rparsing log, completed traces ::   3%|3         | 960/31509 [00:02<00:52, 586.16it/s]","\rparsing log, completed traces ::   3%|3         | 1020/31509 [00:02<01:27, 347.99it/s]","\rparsing log, completed traces ::   3%|3         | 1070/31509 [00:02<01:20, 377.72it/s]","\rparsing log, completed traces ::   4%|3         | 1129/31509 [00:03<01:11, 424.70it/s]","\rparsing log, completed traces ::   4%|3         | 1194/31509 [00:03<01:03, 476.90it/s]","\rparsing log, completed traces ::   4%|3         | 1258/31509 [00:03<00:58, 517.41it/s]","\rparsing log, completed traces ::   4%|4         | 1316/31509 [00:03<00:57, 521.68it/s]","\rparsing log, completed traces ::   4%|4         | 1383/31509 [00:03<00:53, 561.69it/s]","\rparsing log, completed traces ::   5%|4         | 1443/31509 [00:03<00:54, 549.13it/s]","\rparsing log, completed traces ::   5%|4         | 1501/31509 [00:03<00:54, 554.99it/s]","\rparsing log, completed traces ::   5%|4         | 1562/31509 [00:03<00:52, 569.68it/s]","\rparsing log, completed traces ::   5%|5         | 1621/31509 [00:03<00:52, 572.91it/s]","\rparsing log, completed traces ::   5%|5         | 1680/31509 [00:04<00:52, 566.00it/s]","\rparsing log, completed traces ::   6%|5         | 1744/31509 [00:04<00:50, 585.20it/s]","\rparsing log, completed traces ::   6%|5         | 1804/31509 [00:04<00:52, 563.49it/s]","\rparsing log, completed traces ::   6%|5         | 1861/31509 [00:04<00:54, 545.28it/s]","\rparsing log, completed traces ::   6%|6         | 1916/31509 [00:04<01:32, 318.77it/s]","\rparsing log, completed traces ::   6%|6         | 1975/31509 [00:04<01:19, 370.10it/s]","\rparsing log, completed traces ::   6%|6         | 2032/31509 [00:04<01:11, 411.30it/s]","\rparsing log, completed traces ::   7%|6         | 2090/31509 [00:04<01:05, 449.81it/s]","\rparsing log, completed traces ::   7%|6         | 2150/31509 [00:05<01:00, 486.01it/s]","\rparsing log, completed traces ::   7%|7         | 2208/31509 [00:05<00:57, 509.57it/s]","\rparsing log, completed traces ::   7%|7         | 2272/31509 [00:05<00:53, 541.85it/s]","\rparsing log, completed traces ::   7%|7         | 2332/31509 [00:05<00:52, 557.57it/s]","\rparsing log, completed traces ::   8%|7         | 2391/31509 [00:05<00:51, 562.53it/s]","\rparsing log, completed traces ::   8%|7         | 2452/31509 [00:05<00:50, 573.98it/s]","\rparsing log, completed traces ::   8%|7         | 2511/31509 [00:05<00:50, 574.70it/s]","\rparsing log, completed traces ::   8%|8         | 2576/31509 [00:05<00:48, 595.20it/s]","\rparsing log, completed traces ::   8%|8         | 2640/31509 [00:05<00:47, 607.39it/s]","\rparsing log, completed traces ::   9%|8         | 2702/31509 [00:06<00:47, 604.54it/s]","\rparsing log, completed traces ::   9%|8         | 2763/31509 [00:06<00:47, 599.63it/s]","\rparsing log, completed traces ::   9%|8         | 2828/31509 [00:06<00:46, 611.36it/s]","\rparsing log, completed traces ::   9%|9         | 2896/31509 [00:06<00:45, 629.31it/s]","\rparsing log, completed traces ::   9%|9         | 2962/31509 [00:06<00:44, 636.52it/s]","\rparsing log, completed traces ::  10%|9         | 3026/31509 [00:06<01:24, 335.54it/s]","\rparsing log, completed traces ::  10%|9         | 3084/31509 [00:06<01:14, 379.43it/s]","\rparsing log, completed traces ::  10%|9         | 3143/31509 [00:07<01:07, 422.59it/s]","\rparsing log, completed traces ::  10%|#         | 3200/31509 [00:07<01:02, 454.62it/s]","\rparsing log, completed traces ::  10%|#         | 3265/31509 [00:07<00:56, 502.18it/s]","\rparsing log, completed traces ::  11%|#         | 3335/31509 [00:07<00:50, 552.85it/s]","\rparsing log, completed traces ::  11%|#         | 3401/31509 [00:07<00:48, 579.36it/s]","\rparsing log, completed traces ::  11%|#1        | 3470/31509 [00:07<00:46, 608.85it/s]","\rparsing log, completed traces ::  11%|#1        | 3535/31509 [00:07<00:45, 613.05it/s]","\rparsing log, completed traces ::  11%|#1        | 3600/31509 [00:07<00:44, 620.64it/s]","\rparsing log, completed traces ::  12%|#1        | 3664/31509 [00:07<00:44, 622.06it/s]","\rparsing log, completed traces ::  12%|#1        | 3728/31509 [00:07<00:44, 625.48it/s]","\rparsing log, completed traces ::  12%|#2        | 3792/31509 [00:08<00:44, 619.92it/s]","\rparsing log, completed traces ::  12%|#2        | 3855/31509 [00:08<00:45, 612.03it/s]","\rparsing log, completed traces ::  12%|#2        | 3919/31509 [00:08<00:44, 618.03it/s]","\rparsing log, completed traces ::  13%|#2        | 3982/31509 [00:08<00:45, 604.79it/s]","\rparsing log, completed traces ::  13%|#2        | 4043/31509 [00:08<00:53, 516.50it/s]","\rparsing log, completed traces ::  13%|#3        | 4105/31509 [00:08<00:50, 542.98it/s]","\rparsing log, completed traces ::  13%|#3        | 4166/31509 [00:08<00:48, 560.00it/s]","\rparsing log, completed traces ::  13%|#3        | 4224/31509 [00:09<01:37, 278.91it/s]","\rparsing log, completed traces ::  14%|#3        | 4283/31509 [00:09<01:22, 328.93it/s]","\rparsing log, completed traces ::  14%|#3        | 4348/31509 [00:09<01:09, 388.88it/s]","\rparsing log, completed traces ::  14%|#4        | 4413/31509 [00:09<01:00, 444.37it/s]","\rparsing log, completed traces ::  14%|#4        | 4470/31509 [00:09<00:57, 473.33it/s]","\rparsing log, completed traces ::  14%|#4        | 4536/31509 [00:09<00:52, 517.47it/s]","\rparsing log, completed traces ::  15%|#4        | 4602/31509 [00:09<00:48, 554.53it/s]","\rparsing log, completed traces ::  15%|#4        | 4669/31509 [00:09<00:45, 584.17it/s]","\rparsing log, completed traces ::  15%|#5        | 4739/31509 [00:09<00:43, 616.49it/s]","\rparsing log, completed traces ::  15%|#5        | 4804/31509 [00:10<00:43, 617.39it/s]","\rparsing log, completed traces ::  15%|#5        | 4870/31509 [00:10<00:42, 628.30it/s]","\rparsing log, completed traces ::  16%|#5        | 4935/31509 [00:10<00:42, 625.14it/s]","\rparsing log, completed traces ::  16%|#5        | 4999/31509 [00:10<00:42, 626.87it/s]","\rparsing log, completed traces ::  16%|#6        | 5069/31509 [00:10<00:40, 646.82it/s]","\rparsing log, completed traces ::  16%|#6        | 5137/31509 [00:10<00:40, 655.32it/s]","\rparsing log, completed traces ::  17%|#6        | 5203/31509 [00:10<00:40, 642.20it/s]","\rparsing log, completed traces ::  17%|#6        | 5269/31509 [00:10<00:40, 646.85it/s]","\rparsing log, completed traces ::  17%|#6        | 5334/31509 [00:10<00:40, 644.92it/s]","\rparsing log, completed traces ::  17%|#7        | 5399/31509 [00:11<00:41, 634.09it/s]","\rparsing log, completed traces ::  17%|#7        | 5467/31509 [00:11<00:40, 645.62it/s]","\rparsing log, completed traces ::  18%|#7        | 5533/31509 [00:11<00:40, 647.57it/s]","\rparsing log, completed traces ::  18%|#7        | 5602/31509 [00:11<00:39, 659.92it/s]","\rparsing log, completed traces ::  18%|#7        | 5669/31509 [00:11<01:24, 306.16it/s]","\rparsing log, completed traces ::  18%|#8        | 5743/31509 [00:11<01:08, 376.21it/s]","\rparsing log, completed traces ::  18%|#8        | 5808/31509 [00:12<01:00, 427.25it/s]","\rparsing log, completed traces ::  19%|#8        | 5877/31509 [00:12<00:53, 482.32it/s]","\rparsing log, completed traces ::  19%|#8        | 5953/31509 [00:12<00:47, 541.69it/s]","\rparsing log, completed traces ::  19%|#9        | 6019/31509 [00:12<00:45, 561.48it/s]","\rparsing log, completed traces ::  19%|#9        | 6084/31509 [00:12<00:44, 577.35it/s]","\rparsing log, completed traces ::  20%|#9        | 6153/31509 [00:12<00:41, 604.19it/s]","\rparsing log, completed traces ::  20%|#9        | 6218/31509 [00:12<00:41, 615.84it/s]","\rparsing log, completed traces ::  20%|#9        | 6283/31509 [00:12<00:40, 623.27it/s]","\rparsing log, completed traces ::  20%|##        | 6348/31509 [00:12<00:40, 626.81it/s]","\rparsing log, completed traces ::  20%|##        | 6416/31509 [00:12<00:39, 640.80it/s]","\rparsing log, completed traces ::  21%|##        | 6482/31509 [00:13<00:39, 635.19it/s]","\rparsing log, completed traces ::  21%|##        | 6547/31509 [00:13<00:40, 611.71it/s]","\rparsing log, completed traces ::  21%|##        | 6609/31509 [00:13<00:41, 601.24it/s]","\rparsing log, completed traces ::  21%|##1       | 6674/31509 [00:13<00:40, 614.87it/s]","\rparsing log, completed traces ::  21%|##1       | 6740/31509 [00:13<00:39, 625.29it/s]","\rparsing log, completed traces ::  22%|##1       | 6803/31509 [00:13<00:39, 626.43it/s]","\rparsing log, completed traces ::  22%|##1       | 6866/31509 [00:13<00:39, 617.47it/s]","\rparsing log, completed traces ::  22%|##2       | 6936/31509 [00:13<00:38, 638.95it/s]","\rparsing log, completed traces ::  22%|##2       | 7001/31509 [00:13<00:38, 633.46it/s]","\rparsing log, completed traces ::  22%|##2       | 7072/31509 [00:13<00:37, 655.79it/s]","\rparsing log, completed traces ::  23%|##2       | 7139/31509 [00:14<00:37, 656.99it/s]","\rparsing log, completed traces ::  23%|##2       | 7205/31509 [00:14<01:23, 290.46it/s]","\rparsing log, completed traces ::  23%|##3       | 7270/31509 [00:14<01:09, 346.72it/s]","\rparsing log, completed traces ::  23%|##3       | 7330/31509 [00:14<01:01, 391.80it/s]","\rparsing log, completed traces ::  23%|##3       | 7393/31509 [00:14<00:54, 440.57it/s]","\rparsing log, completed traces ::  24%|##3       | 7454/31509 [00:15<00:50, 478.97it/s]","\rparsing log, completed traces ::  24%|##3       | 7528/31509 [00:15<00:44, 541.05it/s]","\rparsing log, completed traces ::  24%|##4       | 7594/31509 [00:15<00:41, 571.76it/s]","\rparsing log, completed traces ::  24%|##4       | 7658/31509 [00:15<00:41, 577.26it/s]","\rparsing log, completed traces ::  25%|##4       | 7722/31509 [00:15<00:40, 593.47it/s]","\rparsing log, completed traces ::  25%|##4       | 7785/31509 [00:15<00:40, 585.35it/s]","\rparsing log, completed traces ::  25%|##4       | 7851/31509 [00:15<00:39, 606.18it/s]","\rparsing log, completed traces ::  25%|##5       | 7917/31509 [00:15<00:38, 620.03it/s]","\rparsing log, completed traces ::  25%|##5       | 7982/31509 [00:15<00:37, 628.53it/s]","\rparsing log, completed traces ::  26%|##5       | 8048/31509 [00:15<00:36, 634.62it/s]","\rparsing log, completed traces ::  26%|##5       | 8113/31509 [00:16<00:37, 630.93it/s]","\rparsing log, completed traces ::  26%|##5       | 8181/31509 [00:16<00:36, 644.84it/s]","\rparsing log, completed traces ::  26%|##6       | 8246/31509 [00:16<00:36, 644.52it/s]","\rparsing log, completed traces ::  26%|##6       | 8314/31509 [00:16<00:35, 653.43it/s]","\rparsing log, completed traces ::  27%|##6       | 8380/31509 [00:16<00:36, 641.55it/s]","\rparsing log, completed traces ::  27%|##6       | 8445/31509 [00:16<00:36, 630.34it/s]","\rparsing log, completed traces ::  27%|##7       | 8511/31509 [00:16<00:36, 638.50it/s]","\rparsing log, completed traces ::  27%|##7       | 8575/31509 [00:16<00:37, 617.55it/s]","\rparsing log, completed traces ::  27%|##7       | 8637/31509 [00:16<00:36, 618.19it/s]","\rparsing log, completed traces ::  28%|##7       | 8701/31509 [00:16<00:36, 622.19it/s]","\rparsing log, completed traces ::  28%|##7       | 8766/31509 [00:17<00:36, 627.61it/s]","\rparsing log, completed traces ::  28%|##8       | 8829/31509 [00:17<00:36, 625.73it/s]","\rparsing log, completed traces ::  28%|##8       | 8892/31509 [00:17<00:36, 617.36it/s]","\rparsing log, completed traces ::  28%|##8       | 8954/31509 [00:17<01:28, 255.45it/s]","\rparsing log, completed traces ::  29%|##8       | 9011/31509 [00:17<01:14, 301.11it/s]","\rparsing log, completed traces ::  29%|##8       | 9069/31509 [00:18<01:04, 349.13it/s]","\rparsing log, completed traces ::  29%|##8       | 9131/31509 [00:18<00:55, 402.30it/s]","\rparsing log, completed traces ::  29%|##9       | 9193/31509 [00:18<00:49, 449.09it/s]","\rparsing log, completed traces ::  29%|##9       | 9250/31509 [00:18<00:46, 475.45it/s]","\rparsing log, completed traces ::  30%|##9       | 9313/31509 [00:18<00:43, 514.50it/s]","\rparsing log, completed traces ::  30%|##9       | 9375/31509 [00:18<00:40, 540.92it/s]","\rparsing log, completed traces ::  30%|##9       | 9435/31509 [00:18<00:40, 547.13it/s]","\rparsing log, completed traces ::  30%|###       | 9498/31509 [00:18<00:38, 565.93it/s]","\rparsing log, completed traces ::  30%|###       | 9562/31509 [00:18<00:37, 584.75it/s]","\rparsing log, completed traces ::  31%|###       | 9623/31509 [00:18<00:37, 590.41it/s]","\rparsing log, completed traces ::  31%|###       | 9684/31509 [00:19<00:37, 588.31it/s]","\rparsing log, completed traces ::  31%|###       | 9744/31509 [00:19<00:37, 583.55it/s]","\rparsing log, completed traces ::  31%|###1      | 9806/31509 [00:19<00:36, 591.23it/s]","\rparsing log, completed traces ::  31%|###1      | 9866/31509 [00:19<00:36, 592.54it/s]","\rparsing log, completed traces ::  32%|###1      | 9926/31509 [00:19<00:36, 591.96it/s]","\rparsing log, completed traces ::  32%|###1      | 9986/31509 [00:19<00:36, 588.87it/s]","\rparsing log, completed traces ::  32%|###1      | 10046/31509 [00:19<00:36, 590.58it/s]","\rparsing log, completed traces ::  32%|###2      | 10106/31509 [00:19<00:37, 567.76it/s]","\rparsing log, completed traces ::  32%|###2      | 10167/31509 [00:19<00:36, 578.72it/s]","\rparsing log, completed traces ::  32%|###2      | 10229/31509 [00:20<00:36, 590.09it/s]","\rparsing log, completed traces ::  33%|###2      | 10290/31509 [00:20<00:35, 594.49it/s]","\rparsing log, completed traces ::  33%|###2      | 10350/31509 [00:20<00:36, 573.63it/s]","\rparsing log, completed traces ::  33%|###3      | 10411/31509 [00:20<00:36, 583.29it/s]","\rparsing log, completed traces ::  33%|###3      | 10470/31509 [00:20<00:36, 577.99it/s]","\rparsing log, completed traces ::  33%|###3      | 10528/31509 [00:20<00:36, 572.62it/s]","\rparsing log, completed traces ::  34%|###3      | 10590/31509 [00:20<00:35, 583.89it/s]","\rparsing log, completed traces ::  34%|###3      | 10649/31509 [00:20<00:36, 571.26it/s]","\rparsing log, completed traces ::  34%|###3      | 10709/31509 [00:20<00:36, 573.43it/s]","\rparsing log, completed traces ::  34%|###4      | 10767/31509 [00:21<01:30, 229.72it/s]","\rparsing log, completed traces ::  34%|###4      | 10821/31509 [00:21<01:15, 273.50it/s]","\rparsing log, completed traces ::  35%|###4      | 10876/31509 [00:21<01:04, 319.38it/s]","\rparsing log, completed traces ::  35%|###4      | 10933/31509 [00:21<00:56, 367.24it/s]","\rparsing log, completed traces ::  35%|###4      | 10994/31509 [00:21<00:48, 419.56it/s]","\rparsing log, completed traces ::  35%|###5      | 11054/31509 [00:21<00:44, 462.13it/s]","\rparsing log, completed traces ::  35%|###5      | 11117/31509 [00:22<00:40, 502.61it/s]","\rparsing log, completed traces ::  35%|###5      | 11179/31509 [00:22<00:38, 528.80it/s]","\rparsing log, completed traces ::  36%|###5      | 11245/31509 [00:22<00:35, 564.11it/s]","\rparsing log, completed traces ::  36%|###5      | 11307/31509 [00:22<00:34, 578.99it/s]","\rparsing log, completed traces ::  36%|###6      | 11368/31509 [00:22<00:35, 570.12it/s]","\rparsing log, completed traces ::  36%|###6      | 11428/31509 [00:22<00:34, 577.03it/s]","\rparsing log, completed traces ::  36%|###6      | 11490/31509 [00:22<00:34, 586.48it/s]","\rparsing log, completed traces ::  37%|###6      | 11550/31509 [00:22<00:34, 578.04it/s]","\rparsing log, completed traces ::  37%|###6      | 11612/31509 [00:22<00:33, 587.35it/s]","\rparsing log, completed traces ::  37%|###7      | 11674/31509 [00:23<00:33, 595.13it/s]","\rparsing log, completed traces ::  37%|###7      | 11736/31509 [00:23<00:32, 602.03it/s]","\rparsing log, completed traces ::  37%|###7      | 11799/31509 [00:23<00:32, 609.10it/s]","\rparsing log, completed traces ::  38%|###7      | 11861/31509 [00:23<00:32, 611.89it/s]","\rparsing log, completed traces ::  38%|###7      | 11928/31509 [00:23<00:31, 627.37it/s]","\rparsing log, completed traces ::  38%|###8      | 11991/31509 [00:23<00:31, 621.78it/s]","\rparsing log, completed traces ::  38%|###8      | 12054/31509 [00:23<00:31, 618.93it/s]","\rparsing log, completed traces ::  38%|###8      | 12116/31509 [00:23<00:31, 614.50it/s]","\rparsing log, completed traces ::  39%|###8      | 12178/31509 [00:23<00:31, 604.52it/s]","\rparsing log, completed traces ::  39%|###8      | 12242/31509 [00:23<00:31, 611.76it/s]","\rparsing log, completed traces ::  39%|###9      | 12304/31509 [00:24<00:31, 607.93it/s]","\rparsing log, completed traces ::  39%|###9      | 12370/31509 [00:24<00:30, 621.03it/s]","\rparsing log, completed traces ::  39%|###9      | 12433/31509 [00:24<00:30, 615.47it/s]","\rparsing log, completed traces ::  40%|###9      | 12496/31509 [00:24<00:30, 619.26it/s]","\rparsing log, completed traces ::  40%|###9      | 12558/31509 [00:24<00:31, 604.98it/s]","\rparsing log, completed traces ::  40%|####      | 12621/31509 [00:24<00:30, 612.24it/s]","\rparsing log, completed traces ::  40%|####      | 12683/31509 [00:24<00:30, 613.55it/s]","\rparsing log, completed traces ::  40%|####      | 12749/31509 [00:24<00:30, 624.28it/s]","\rparsing log, completed traces ::  41%|####      | 12812/31509 [00:25<01:24, 221.71it/s]","\rparsing log, completed traces ::  41%|####      | 12867/31509 [00:25<01:10, 264.34it/s]","\rparsing log, completed traces ::  41%|####1     | 12927/31509 [00:25<00:58, 316.71it/s]","\rparsing log, completed traces ::  41%|####1     | 12993/31509 [00:25<00:48, 377.96it/s]","\rparsing log, completed traces ::  41%|####1     | 13059/31509 [00:25<00:42, 434.70it/s]","\rparsing log, completed traces ::  42%|####1     | 13125/31509 [00:25<00:37, 484.76it/s]","\rparsing log, completed traces ::  42%|####1     | 13186/31509 [00:26<00:35, 509.23it/s]","\rparsing log, completed traces ::  42%|####2     | 13247/31509 [00:26<00:34, 534.69it/s]","\rparsing log, completed traces ::  42%|####2     | 13309/31509 [00:26<00:32, 556.50it/s]","\rparsing log, completed traces ::  42%|####2     | 13371/31509 [00:26<00:31, 573.83it/s]","\rparsing log, completed traces ::  43%|####2     | 13433/31509 [00:26<00:30, 583.76it/s]","\rparsing log, completed traces ::  43%|####2     | 13497/31509 [00:26<00:30, 596.97it/s]","\rparsing log, completed traces ::  43%|####3     | 13559/31509 [00:26<00:30, 596.59it/s]","\rparsing log, completed traces ::  43%|####3     | 13620/31509 [00:26<00:30, 591.73it/s]","\rparsing log, completed traces ::  43%|####3     | 13681/31509 [00:26<00:30, 589.15it/s]","\rparsing log, completed traces ::  44%|####3     | 13746/31509 [00:27<00:29, 603.87it/s]","\rparsing log, completed traces ::  44%|####3     | 13807/31509 [00:27<00:29, 604.84it/s]","\rparsing log, completed traces ::  44%|####4     | 13870/31509 [00:27<00:29, 608.24it/s]","\rparsing log, completed traces ::  44%|####4     | 13932/31509 [00:27<00:28, 609.07it/s]","\rparsing log, completed traces ::  44%|####4     | 13994/31509 [00:27<00:28, 609.55it/s]","\rparsing log, completed traces ::  45%|####4     | 14060/31509 [00:27<00:27, 623.98it/s]","\rparsing log, completed traces ::  45%|####4     | 14123/31509 [00:27<00:27, 620.94it/s]","\rparsing log, completed traces ::  45%|####5     | 14186/31509 [00:27<00:27, 621.53it/s]","\rparsing log, completed traces ::  45%|####5     | 14250/31509 [00:27<00:27, 622.90it/s]","\rparsing log, completed traces ::  45%|####5     | 14314/31509 [00:27<00:27, 627.06it/s]","\rparsing log, completed traces ::  46%|####5     | 14382/31509 [00:28<00:26, 642.21it/s]","\rparsing log, completed traces ::  46%|####5     | 14447/31509 [00:28<00:26, 640.87it/s]","\rparsing log, completed traces ::  46%|####6     | 14512/31509 [00:28<00:26, 629.83it/s]","\rparsing log, completed traces ::  46%|####6     | 14576/31509 [00:28<00:27, 605.47it/s]","\rparsing log, completed traces ::  46%|####6     | 14638/31509 [00:28<00:27, 608.21it/s]","\rparsing log, completed traces ::  47%|####6     | 14702/31509 [00:28<00:27, 616.94it/s]","\rparsing log, completed traces ::  47%|####6     | 14766/31509 [00:28<00:26, 620.90it/s]","\rparsing log, completed traces ::  47%|####7     | 14834/31509 [00:28<00:26, 638.13it/s]","\rparsing log, completed traces ::  47%|####7     | 14898/31509 [00:28<00:26, 631.08it/s]","\rparsing log, completed traces ::  48%|####7     | 14972/31509 [00:28<00:25, 659.03it/s]","\rparsing log, completed traces ::  48%|####7     | 15038/31509 [00:29<00:25, 651.15it/s]","\rparsing log, completed traces ::  48%|####7     | 15110/31509 [00:29<00:24, 670.64it/s]","\rparsing log, completed traces ::  48%|####8     | 15179/31509 [00:29<00:24, 673.39it/s]","\rparsing log, completed traces ::  48%|####8     | 15247/31509 [00:30<01:14, 219.73it/s]","\rparsing log, completed traces ::  49%|####8     | 15314/31509 [00:30<00:59, 273.79it/s]","\rparsing log, completed traces ::  49%|####8     | 15379/31509 [00:30<00:49, 328.92it/s]","\rparsing log, completed traces ::  49%|####9     | 15446/31509 [00:30<00:41, 387.87it/s]","\rparsing log, completed traces ::  49%|####9     | 15509/31509 [00:30<00:36, 435.27it/s]","\rparsing log, completed traces ::  49%|####9     | 15578/31509 [00:30<00:32, 491.02it/s]","\rparsing log, completed traces ::  50%|####9     | 15642/31509 [00:30<00:30, 525.60it/s]","\rparsing log, completed traces ::  50%|####9     | 15708/31509 [00:30<00:28, 558.42it/s]","\rparsing log, completed traces ::  50%|#####     | 15772/31509 [00:30<00:27, 570.33it/s]","\rparsing log, completed traces ::  50%|#####     | 15835/31509 [00:30<00:26, 581.61it/s]","\rparsing log, completed traces ::  50%|#####     | 15903/31509 [00:31<00:25, 605.58it/s]","\rparsing log, completed traces ::  51%|#####     | 15967/31509 [00:31<00:25, 614.22it/s]","\rparsing log, completed traces ::  51%|#####     | 16031/31509 [00:31<00:25, 610.28it/s]","\rparsing log, completed traces ::  51%|#####1    | 16098/31509 [00:31<00:24, 627.05it/s]","\rparsing log, completed traces ::  51%|#####1    | 16168/31509 [00:31<00:23, 647.11it/s]","\rparsing log, completed traces ::  52%|#####1    | 16234/31509 [00:31<00:23, 650.13it/s]","\rparsing log, completed traces ::  52%|#####1    | 16301/31509 [00:31<00:23, 655.13it/s]","\rparsing log, completed traces ::  52%|#####1    | 16367/31509 [00:31<00:24, 623.27it/s]","\rparsing log, completed traces ::  52%|#####2    | 16430/31509 [00:31<00:24, 617.10it/s]","\rparsing log, completed traces ::  52%|#####2    | 16493/31509 [00:32<00:24, 604.99it/s]","\rparsing log, completed traces ::  53%|#####2    | 16557/31509 [00:32<00:24, 612.81it/s]","\rparsing log, completed traces ::  53%|#####2    | 16620/31509 [00:32<00:24, 615.64it/s]","\rparsing log, completed traces ::  53%|#####2    | 16689/31509 [00:32<00:23, 636.45it/s]","\rparsing log, completed traces ::  53%|#####3    | 16753/31509 [00:32<00:23, 619.97it/s]","\rparsing log, completed traces ::  53%|#####3    | 16820/31509 [00:32<00:23, 630.89it/s]","\rparsing log, completed traces ::  54%|#####3    | 16884/31509 [00:32<00:23, 631.85it/s]","\rparsing log, completed traces ::  54%|#####3    | 16949/31509 [00:32<00:22, 634.58it/s]","\rparsing log, completed traces ::  54%|#####3    | 17014/31509 [00:32<00:22, 638.95it/s]","\rparsing log, completed traces ::  54%|#####4    | 17087/31509 [00:32<00:21, 665.81it/s]","\rparsing log, completed traces ::  54%|#####4    | 17154/31509 [00:33<00:22, 650.78it/s]","\rparsing log, completed traces ::  55%|#####4    | 17220/31509 [00:33<00:22, 634.29it/s]","\rparsing log, completed traces ::  55%|#####4    | 17286/31509 [00:33<00:22, 639.46it/s]","\rparsing log, completed traces ::  55%|#####5    | 17351/31509 [00:33<00:22, 630.52it/s]","\rparsing log, completed traces ::  55%|#####5    | 17420/31509 [00:33<00:21, 644.78it/s]","\rparsing log, completed traces ::  55%|#####5    | 17485/31509 [00:33<00:22, 626.30it/s]","\rparsing log, completed traces ::  56%|#####5    | 17548/31509 [00:33<00:22, 623.10it/s]","\rparsing log, completed traces ::  56%|#####5    | 17618/31509 [00:33<00:21, 643.46it/s]","\rparsing log, completed traces ::  56%|#####6    | 17683/31509 [00:33<00:21, 641.60it/s]","\rparsing log, completed traces ::  56%|#####6    | 17748/31509 [00:33<00:21, 635.11it/s]","\rparsing log, completed traces ::  57%|#####6    | 17815/31509 [00:34<00:21, 643.98it/s]","\rparsing log, completed traces ::  57%|#####6    | 17881/31509 [00:34<00:21, 645.87it/s]","\rparsing log, completed traces ::  57%|#####6    | 17948/31509 [00:34<00:20, 652.37it/s]","\rparsing log, completed traces ::  57%|#####7    | 18015/31509 [00:34<00:20, 656.02it/s]","\rparsing log, completed traces ::  57%|#####7    | 18081/31509 [00:35<01:04, 209.73it/s]","\rparsing log, completed traces ::  58%|#####7    | 18153/31509 [00:35<00:49, 270.35it/s]","\rparsing log, completed traces ::  58%|#####7    | 18220/31509 [00:35<00:40, 328.22it/s]","\rparsing log, completed traces ::  58%|#####8    | 18292/31509 [00:35<00:33, 394.50it/s]","\rparsing log, completed traces ::  58%|#####8    | 18359/31509 [00:35<00:29, 448.13it/s]","\rparsing log, completed traces ::  58%|#####8    | 18425/31509 [00:35<00:26, 494.30it/s]","\rparsing log, completed traces ::  59%|#####8    | 18490/31509 [00:35<00:24, 530.71it/s]","\rparsing log, completed traces ::  59%|#####8    | 18555/31509 [00:35<00:23, 542.34it/s]","\rparsing log, completed traces ::  59%|#####9    | 18621/31509 [00:36<00:22, 569.14it/s]","\rparsing log, completed traces ::  59%|#####9    | 18697/31509 [00:36<00:20, 618.18it/s]","\rparsing log, completed traces ::  60%|#####9    | 18765/31509 [00:36<00:20, 632.44it/s]","\rparsing log, completed traces ::  60%|#####9    | 18832/31509 [00:36<00:20, 630.26it/s]","\rparsing log, completed traces ::  60%|#####9    | 18901/31509 [00:36<00:19, 646.84it/s]","\rparsing log, completed traces ::  60%|######    | 18968/31509 [00:36<00:19, 646.21it/s]","\rparsing log, completed traces ::  60%|######    | 19034/31509 [00:36<00:19, 647.62it/s]","\rparsing log, completed traces ::  61%|######    | 19100/31509 [00:36<00:19, 650.82it/s]","\rparsing log, completed traces ::  61%|######    | 19168/31509 [00:36<00:18, 659.34it/s]","\rparsing log, completed traces ::  61%|######1   | 19235/31509 [00:36<00:19, 638.57it/s]","\rparsing log, completed traces ::  61%|######1   | 19302/31509 [00:37<00:18, 647.11it/s]","\rparsing log, completed traces ::  61%|######1   | 19369/31509 [00:37<00:18, 651.24it/s]","\rparsing log, completed traces ::  62%|######1   | 19440/31509 [00:37<00:18, 664.09it/s]","\rparsing log, completed traces ::  62%|######1   | 19507/31509 [00:37<00:18, 657.90it/s]","\rparsing log, completed traces ::  62%|######2   | 19575/31509 [00:37<00:17, 664.31it/s]","\rparsing log, completed traces ::  62%|######2   | 19642/31509 [00:37<00:18, 658.53it/s]","\rparsing log, completed traces ::  63%|######2   | 19708/31509 [00:37<00:18, 637.31it/s]","\rparsing log, completed traces ::  63%|######2   | 19772/31509 [00:37<00:18, 631.06it/s]","\rparsing log, completed traces ::  63%|######2   | 19836/31509 [00:37<00:18, 631.46it/s]","\rparsing log, completed traces ::  63%|######3   | 19900/31509 [00:37<00:18, 621.44it/s]","\rparsing log, completed traces ::  63%|######3   | 19964/31509 [00:38<00:18, 625.76it/s]","\rparsing log, completed traces ::  64%|######3   | 20028/31509 [00:38<00:18, 629.15it/s]","\rparsing log, completed traces ::  64%|######3   | 20091/31509 [00:38<00:18, 623.85it/s]","\rparsing log, completed traces ::  64%|######3   | 20156/31509 [00:38<00:18, 630.51it/s]","\rparsing log, completed traces ::  64%|######4   | 20225/31509 [00:38<00:17, 647.85it/s]","\rparsing log, completed traces ::  64%|######4   | 20295/31509 [00:38<00:16, 662.40it/s]","\rparsing log, completed traces ::  65%|######4   | 20362/31509 [00:38<00:17, 646.74it/s]","\rparsing log, completed traces ::  65%|######4   | 20437/31509 [00:38<00:16, 674.79it/s]","\rparsing log, completed traces ::  65%|######5   | 20505/31509 [00:38<00:16, 671.54it/s]","\rparsing log, completed traces ::  65%|######5   | 20573/31509 [00:39<00:16, 667.85it/s]","\rparsing log, completed traces ::  66%|######5   | 20648/31509 [00:39<00:15, 689.14it/s]","\rparsing log, completed traces ::  66%|######5   | 20717/31509 [00:39<00:15, 689.03it/s]","\rparsing log, completed traces ::  66%|######5   | 20786/31509 [00:39<00:15, 683.66it/s]","\rparsing log, completed traces ::  66%|######6   | 20855/31509 [00:39<00:15, 679.15it/s]","\rparsing log, completed traces ::  66%|######6   | 20923/31509 [00:39<00:16, 659.74it/s]","\rparsing log, completed traces ::  67%|######6   | 20990/31509 [00:39<00:16, 653.74it/s]","\rparsing log, completed traces ::  67%|######6   | 21056/31509 [00:39<00:16, 639.64it/s]","\rparsing log, completed traces ::  67%|######7   | 21123/31509 [00:39<00:16, 648.21it/s]","\rparsing log, completed traces ::  67%|######7   | 21188/31509 [00:39<00:16, 640.32it/s]","\rparsing log, completed traces ::  67%|######7   | 21258/31509 [00:40<00:15, 655.01it/s]","\rparsing log, completed traces ::  68%|######7   | 21324/31509 [00:40<00:53, 191.96it/s]","\rparsing log, completed traces ::  68%|######7   | 21385/31509 [00:41<00:42, 237.45it/s]","\rparsing log, completed traces ::  68%|######8   | 21458/31509 [00:41<00:33, 303.53it/s]","\rparsing log, completed traces ::  68%|######8   | 21528/31509 [00:41<00:27, 366.62it/s]","\rparsing log, completed traces ::  69%|######8   | 21598/31509 [00:41<00:23, 427.02it/s]","\rparsing log, completed traces ::  69%|######8   | 21662/31509 [00:41<00:21, 468.90it/s]","\rparsing log, completed traces ::  69%|######8   | 21733/31509 [00:41<00:18, 524.27it/s]","\rparsing log, completed traces ::  69%|######9   | 21801/31509 [00:41<00:17, 560.92it/s]","\rparsing log, completed traces ::  69%|######9   | 21876/31509 [00:41<00:15, 609.19it/s]","\rparsing log, completed traces ::  70%|######9   | 21945/31509 [00:41<00:15, 623.46it/s]","\rparsing log, completed traces ::  70%|######9   | 22013/31509 [00:41<00:14, 638.38it/s]","\rparsing log, completed traces ::  70%|#######   | 22084/31509 [00:42<00:14, 656.01it/s]","\rparsing log, completed traces ::  70%|#######   | 22159/31509 [00:42<00:13, 681.56it/s]","\rparsing log, completed traces ::  71%|#######   | 22230/31509 [00:42<00:13, 670.54it/s]","\rparsing log, completed traces ::  71%|#######   | 22299/31509 [00:42<00:13, 674.79it/s]","\rparsing log, completed traces ::  71%|#######   | 22368/31509 [00:42<00:13, 668.80it/s]","\rparsing log, completed traces ::  71%|#######1  | 22441/31509 [00:42<00:13, 686.37it/s]","\rparsing log, completed traces ::  71%|#######1  | 22512/31509 [00:42<00:13, 691.51it/s]","\rparsing log, completed traces ::  72%|#######1  | 22582/31509 [00:42<00:13, 677.02it/s]","\rparsing log, completed traces ::  72%|#######1  | 22651/31509 [00:42<00:13, 673.65it/s]","\rparsing log, completed traces ::  72%|#######2  | 22720/31509 [00:43<00:13, 672.33it/s]","\rparsing log, completed traces ::  72%|#######2  | 22792/31509 [00:43<00:12, 685.45it/s]","\rparsing log, completed traces ::  73%|#######2  | 22861/31509 [00:43<00:12, 672.82it/s]","\rparsing log, completed traces ::  73%|#######2  | 22929/31509 [00:43<00:13, 653.22it/s]","\rparsing log, completed traces ::  73%|#######2  | 22995/31509 [00:43<00:13, 649.32it/s]","\rparsing log, completed traces ::  73%|#######3  | 23062/31509 [00:43<00:12, 652.39it/s]","\rparsing log, completed traces ::  73%|#######3  | 23128/31509 [00:43<00:12, 647.66it/s]","\rparsing log, completed traces ::  74%|#######3  | 23193/31509 [00:43<00:12, 641.96it/s]","\rparsing log, completed traces ::  74%|#######3  | 23258/31509 [00:43<00:12, 637.82it/s]","\rparsing log, completed traces ::  74%|#######4  | 23324/31509 [00:43<00:12, 642.55it/s]","\rparsing log, completed traces ::  74%|#######4  | 23389/31509 [00:44<00:12, 639.71it/s]","\rparsing log, completed traces ::  74%|#######4  | 23455/31509 [00:44<00:12, 641.40it/s]","\rparsing log, completed traces ::  75%|#######4  | 23520/31509 [00:44<00:12, 643.30it/s]","\rparsing log, completed traces ::  75%|#######4  | 23588/31509 [00:44<00:12, 654.06it/s]","\rparsing log, completed traces ::  75%|#######5  | 23655/31509 [00:44<00:11, 656.67it/s]","\rparsing log, completed traces ::  75%|#######5  | 23721/31509 [00:44<00:11, 653.34it/s]","\rparsing log, completed traces ::  75%|#######5  | 23787/31509 [00:44<00:11, 653.72it/s]","\rparsing log, completed traces ::  76%|#######5  | 23853/31509 [00:44<00:11, 654.73it/s]","\rparsing log, completed traces ::  76%|#######5  | 23919/31509 [00:44<00:11, 651.27it/s]","\rparsing log, completed traces ::  76%|#######6  | 23988/31509 [00:44<00:11, 662.41it/s]","\rparsing log, completed traces ::  76%|#######6  | 24055/31509 [00:45<00:11, 649.90it/s]","\rparsing log, completed traces ::  77%|#######6  | 24121/31509 [00:45<00:11, 650.66it/s]","\rparsing log, completed traces ::  77%|#######6  | 24187/31509 [00:45<00:11, 646.05it/s]","\rparsing log, completed traces ::  77%|#######6  | 24253/31509 [00:45<00:11, 646.83it/s]","\rparsing log, completed traces ::  77%|#######7  | 24320/31509 [00:45<00:11, 653.46it/s]","\rparsing log, completed traces ::  77%|#######7  | 24389/31509 [00:45<00:10, 663.57it/s]","\rparsing log, completed traces ::  78%|#######7  | 24456/31509 [00:45<00:10, 648.41it/s]","\rparsing log, completed traces ::  78%|#######7  | 24524/31509 [00:45<00:10, 656.57it/s]","\rparsing log, completed traces ::  78%|#######8  | 24594/31509 [00:45<00:10, 668.89it/s]","\rparsing log, completed traces ::  78%|#######8  | 24663/31509 [00:45<00:10, 672.89it/s]","\rparsing log, completed traces ::  78%|#######8  | 24731/31509 [00:46<00:10, 671.81it/s]","\rparsing log, completed traces ::  79%|#######8  | 24799/31509 [00:46<00:10, 670.50it/s]","\rparsing log, completed traces ::  79%|#######8  | 24869/31509 [00:46<00:09, 677.31it/s]","\rparsing log, completed traces ::  79%|#######9  | 24937/31509 [00:47<00:35, 183.22it/s]","\rparsing log, completed traces ::  79%|#######9  | 25011/31509 [00:47<00:27, 240.23it/s]","\rparsing log, completed traces ::  80%|#######9  | 25069/31509 [00:47<00:22, 283.44it/s]","\rparsing log, completed traces ::  80%|#######9  | 25137/31509 [00:47<00:18, 344.24it/s]","\rparsing log, completed traces ::  80%|########  | 25209/31509 [00:47<00:15, 411.10it/s]","\rparsing log, completed traces ::  80%|########  | 25273/31509 [00:47<00:13, 452.63it/s]","\rparsing log, completed traces ::  80%|########  | 25336/31509 [00:47<00:12, 482.29it/s]","\rparsing log, completed traces ::  81%|########  | 25410/31509 [00:48<00:11, 543.73it/s]","\rparsing log, completed traces ::  81%|########  | 25478/31509 [00:48<00:10, 576.92it/s]","\rparsing log, completed traces ::  81%|########1 | 25544/31509 [00:48<00:10, 590.32it/s]","\rparsing log, completed traces ::  81%|########1 | 25613/31509 [00:48<00:09, 612.64it/s]","\rparsing log, completed traces ::  81%|########1 | 25679/31509 [00:48<00:09, 618.74it/s]","\rparsing log, completed traces ::  82%|########1 | 25745/31509 [00:48<00:09, 628.76it/s]","\rparsing log, completed traces ::  82%|########1 | 25810/31509 [00:48<00:09, 628.57it/s]","\rparsing log, completed traces ::  82%|########2 | 25875/31509 [00:48<00:08, 630.61it/s]","\rparsing log, completed traces ::  82%|########2 | 25941/31509 [00:48<00:08, 636.06it/s]","\rparsing log, completed traces ::  83%|########2 | 26006/31509 [00:48<00:08, 624.56it/s]","\rparsing log, completed traces ::  83%|########2 | 26070/31509 [00:49<00:08, 621.55it/s]","\rparsing log, completed traces ::  83%|########2 | 26136/31509 [00:49<00:08, 628.85it/s]","\rparsing log, completed traces ::  83%|########3 | 26203/31509 [00:49<00:08, 637.42it/s]","\rparsing log, completed traces ::  83%|########3 | 26269/31509 [00:49<00:08, 643.17it/s]","\rparsing log, completed traces ::  84%|########3 | 26336/31509 [00:49<00:07, 650.69it/s]","\rparsing log, completed traces ::  84%|########3 | 26402/31509 [00:49<00:07, 652.47it/s]","\rparsing log, completed traces ::  84%|########4 | 26468/31509 [00:49<00:07, 647.09it/s]","\rparsing log, completed traces ::  84%|########4 | 26533/31509 [00:49<00:07, 630.82it/s]","\rparsing log, completed traces ::  84%|########4 | 26597/31509 [00:49<00:07, 626.22it/s]","\rparsing log, completed traces ::  85%|########4 | 26660/31509 [00:49<00:07, 624.99it/s]","\rparsing log, completed traces ::  85%|########4 | 26728/31509 [00:50<00:07, 640.45it/s]","\rparsing log, completed traces ::  85%|########5 | 26793/31509 [00:50<00:07, 636.38it/s]","\rparsing log, completed traces ::  85%|########5 | 26857/31509 [00:50<00:07, 634.99it/s]","\rparsing log, completed traces ::  85%|########5 | 26921/31509 [00:50<00:07, 629.87it/s]","\rparsing log, completed traces ::  86%|########5 | 26985/31509 [00:50<00:07, 625.76it/s]","\rparsing log, completed traces ::  86%|########5 | 27048/31509 [00:50<00:07, 612.78it/s]","\rparsing log, completed traces ::  86%|########6 | 27113/31509 [00:50<00:07, 622.57it/s]","\rparsing log, completed traces ::  86%|########6 | 27180/31509 [00:50<00:06, 635.73it/s]","\rparsing log, completed traces ::  86%|########6 | 27247/31509 [00:50<00:06, 645.35it/s]","\rparsing log, completed traces ::  87%|########6 | 27312/31509 [00:51<00:06, 637.02it/s]","\rparsing log, completed traces ::  87%|########6 | 27376/31509 [00:51<00:06, 633.39it/s]","\rparsing log, completed traces ::  87%|########7 | 27441/31509 [00:51<00:06, 635.83it/s]","\rparsing log, completed traces ::  87%|########7 | 27505/31509 [00:51<00:06, 633.68it/s]","\rparsing log, completed traces ::  88%|########7 | 27572/31509 [00:51<00:06, 644.14it/s]","\rparsing log, completed traces ::  88%|########7 | 27638/31509 [00:51<00:05, 648.10it/s]","\rparsing log, completed traces ::  88%|########7 | 27707/31509 [00:51<00:05, 658.06it/s]","\rparsing log, completed traces ::  88%|########8 | 27778/31509 [00:51<00:05, 670.86it/s]","\rparsing log, completed traces ::  88%|########8 | 27846/31509 [00:51<00:05, 656.83it/s]","\rparsing log, completed traces ::  89%|########8 | 27912/31509 [00:51<00:05, 653.59it/s]","\rparsing log, completed traces ::  89%|########8 | 27978/31509 [00:52<00:05, 635.91it/s]","\rparsing log, completed traces ::  89%|########9 | 28044/31509 [00:52<00:05, 639.31it/s]","\rparsing log, completed traces ::  89%|########9 | 28109/31509 [00:52<00:05, 627.19it/s]","\rparsing log, completed traces ::  89%|########9 | 28172/31509 [00:52<00:05, 627.70it/s]","\rparsing log, completed traces ::  90%|########9 | 28240/31509 [00:52<00:05, 640.56it/s]","\rparsing log, completed traces ::  90%|########9 | 28305/31509 [00:52<00:05, 639.28it/s]","\rparsing log, completed traces ::  90%|######### | 28371/31509 [00:52<00:04, 643.41it/s]","\rparsing log, completed traces ::  90%|######### | 28436/31509 [00:52<00:04, 629.47it/s]","\rparsing log, completed traces ::  90%|######### | 28503/31509 [00:52<00:04, 640.41it/s]","\rparsing log, completed traces ::  91%|######### | 28573/31509 [00:52<00:04, 655.63it/s]","\rparsing log, completed traces ::  91%|######### | 28639/31509 [00:53<00:04, 653.14it/s]","\rparsing log, completed traces ::  91%|#########1| 28705/31509 [00:53<00:04, 634.39it/s]","\rparsing log, completed traces ::  91%|#########1| 28769/31509 [00:53<00:04, 607.60it/s]","\rparsing log, completed traces ::  92%|#########1| 28831/31509 [00:53<00:04, 595.42it/s]","\rparsing log, completed traces ::  92%|#########1| 28891/31509 [00:54<00:17, 150.42it/s]","\rparsing log, completed traces ::  92%|#########1| 28956/31509 [00:54<00:13, 196.36it/s]","\rparsing log, completed traces ::  92%|#########2| 29021/31509 [00:54<00:09, 249.13it/s]","\rparsing log, completed traces ::  92%|#########2| 29083/31509 [00:54<00:08, 301.27it/s]","\rparsing log, completed traces ::  92%|#########2| 29144/31509 [00:54<00:06, 353.26it/s]","\rparsing log, completed traces ::  93%|#########2| 29213/31509 [00:55<00:05, 417.44it/s]","\rparsing log, completed traces ::  93%|#########2| 29275/31509 [00:55<00:04, 460.84it/s]","\rparsing log, completed traces ::  93%|#########3| 29339/31509 [00:55<00:04, 502.53it/s]","\rparsing log, completed traces ::  93%|#########3| 29401/31509 [00:55<00:04, 523.30it/s]","\rparsing log, completed traces ::  94%|#########3| 29462/31509 [00:55<00:03, 538.61it/s]","\rparsing log, completed traces ::  94%|#########3| 29522/31509 [00:55<00:03, 542.08it/s]","\rparsing log, completed traces ::  94%|#########3| 29581/31509 [00:55<00:03, 553.58it/s]","\rparsing log, completed traces ::  94%|#########4| 29649/31509 [00:55<00:03, 588.68it/s]","\rparsing log, completed traces ::  94%|#########4| 29717/31509 [00:55<00:02, 611.90it/s]","\rparsing log, completed traces ::  95%|#########4| 29780/31509 [00:55<00:02, 599.33it/s]","\rparsing log, completed traces ::  95%|#########4| 29842/31509 [00:56<00:02, 599.73it/s]","\rparsing log, completed traces ::  95%|#########4| 29907/31509 [00:56<00:02, 613.54it/s]","\rparsing log, completed traces ::  95%|#########5| 29970/31509 [00:56<00:02, 616.66it/s]","\rparsing log, completed traces ::  95%|#########5| 30034/31509 [00:56<00:02, 621.89it/s]","\rparsing log, completed traces ::  96%|#########5| 30098/31509 [00:56<00:02, 623.39it/s]","\rparsing log, completed traces ::  96%|#########5| 30166/31509 [00:56<00:02, 633.91it/s]","\rparsing log, completed traces ::  96%|#########5| 30232/31509 [00:56<00:01, 640.71it/s]","\rparsing log, completed traces ::  96%|#########6| 30297/31509 [00:56<00:01, 641.05it/s]","\rparsing log, completed traces ::  96%|#########6| 30366/31509 [00:56<00:01, 654.60it/s]","\rparsing log, completed traces ::  97%|#########6| 30432/31509 [00:57<00:01, 649.19it/s]","\rparsing log, completed traces ::  97%|#########6| 30497/31509 [00:57<00:01, 641.78it/s]","\rparsing log, completed traces ::  97%|#########7| 30564/31509 [00:57<00:01, 647.23it/s]","\rparsing log, completed traces ::  97%|#########7| 30633/31509 [00:57<00:01, 657.64it/s]","\rparsing log, completed traces ::  97%|#########7| 30699/31509 [00:57<00:01, 652.11it/s]","\rparsing log, completed traces ::  98%|#########7| 30765/31509 [00:57<00:01, 647.60it/s]","\rparsing log, completed traces ::  98%|#########7| 30832/31509 [00:57<00:01, 653.19it/s]","\rparsing log, completed traces ::  98%|#########8| 30898/31509 [00:57<00:00, 646.00it/s]","\rparsing log, completed traces ::  98%|#########8| 30964/31509 [00:57<00:00, 647.99it/s]","\rparsing log, completed traces ::  98%|#########8| 31030/31509 [00:57<00:00, 649.24it/s]","\rparsing log, completed traces ::  99%|#########8| 31099/31509 [00:58<00:00, 660.22it/s]","\rparsing log, completed traces ::  99%|#########8| 31167/31509 [00:58<00:00, 664.45it/s]","\rparsing log, completed traces ::  99%|#########9| 31234/31509 [00:58<00:00, 656.70it/s]","\rparsing log, completed traces ::  99%|#########9| 31300/31509 [00:58<00:00, 616.20it/s]","\rparsing log, completed traces :: 100%|#########9| 31363/31509 [00:58<00:00, 605.56it/s]","\rparsing log, completed traces :: 100%|#########9| 31429/31509 [00:58<00:00, 619.21it/s]","\rparsing log, completed traces :: 100%|#########9| 31492/31509 [00:58<00:00, 606.93it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:58<00:00, 536.74it/s]","\n","[data] Loading XES: /workspace/data/Road_Traffic_Fine_Management_Process.xes","\n","\rparsing log, completed traces ::   0%|          | 0/150370 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 697/150370 [00:00<00:21, 6955.68it/s]","\rparsing log, completed traces ::   1%|          | 1474/150370 [00:00<00:20, 7430.01it/s]","\rparsing log, completed traces ::   1%|1         | 2236/150370 [00:00<00:19, 7515.73it/s]","\rparsing log, completed traces ::   2%|1         | 2990/150370 [00:00<00:19, 7509.73it/s]","\rparsing log, completed traces ::   2%|2         | 3741/150370 [00:00<00:19, 7470.33it/s]","\rparsing log, completed traces ::   3%|2         | 4502/150370 [00:00<00:19, 7515.69it/s]","\rparsing log, completed traces ::   3%|3         | 5254/150370 [00:00<00:32, 4430.50it/s]","\rparsing log, completed traces ::   4%|3         | 6003/150370 [00:01<00:28, 5080.17it/s]","\rparsing log, completed traces ::   4%|4         | 6751/150370 [00:01<00:25, 5642.98it/s]","\rparsing log, completed traces ::   5%|4         | 7484/150370 [00:01<00:23, 6066.06it/s]","\rparsing log, completed traces ::   6%|5         | 8279/150370 [00:01<00:21, 6562.73it/s]","\rparsing log, completed traces ::   6%|6         | 9085/150370 [00:01<00:20, 6971.01it/s]","\rparsing log, completed traces ::   7%|6         | 9847/150370 [00:01<00:19, 7146.43it/s]","\rparsing log, completed traces ::   7%|7         | 10607/150370 [00:01<00:19, 7271.62it/s]","\rparsing log, completed traces ::   8%|7         | 11362/150370 [00:01<00:18, 7351.05it/s]","\rparsing log, completed traces ::   8%|8         | 12116/150370 [00:01<00:18, 7389.10it/s]","\rparsing log, completed traces ::   9%|8         | 12868/150370 [00:01<00:18, 7341.78it/s]","\rparsing log, completed traces ::   9%|9         | 13612/150370 [00:02<00:18, 7357.51it/s]","\rparsing log, completed traces ::  10%|9         | 14355/150370 [00:02<00:32, 4195.11it/s]","\rparsing log, completed traces ::  10%|#         | 15123/150370 [00:02<00:27, 4860.19it/s]","\rparsing log, completed traces ::  11%|#         | 15896/150370 [00:02<00:24, 5479.89it/s]","\rparsing log, completed traces ::  11%|#1        | 16675/150370 [00:02<00:22, 6022.83it/s]","\rparsing log, completed traces ::  12%|#1        | 17465/150370 [00:02<00:20, 6495.07it/s]","\rparsing log, completed traces ::  12%|#2        | 18223/150370 [00:02<00:19, 6761.73it/s]","\rparsing log, completed traces ::  13%|#2        | 19001/150370 [00:02<00:18, 7039.38it/s]","\rparsing log, completed traces ::  13%|#3        | 19755/150370 [00:03<00:18, 7179.54it/s]","\rparsing log, completed traces ::  14%|#3        | 20531/150370 [00:03<00:17, 7344.73it/s]","\rparsing log, completed traces ::  14%|#4        | 21344/150370 [00:03<00:17, 7572.21it/s]","\rparsing log, completed traces ::  15%|#4        | 22164/150370 [00:03<00:16, 7755.87it/s]","\rparsing log, completed traces ::  15%|#5        | 22987/150370 [00:03<00:16, 7893.89it/s]","\rparsing log, completed traces ::  16%|#5        | 23786/150370 [00:03<00:16, 7831.63it/s]","\rparsing log, completed traces ::  16%|#6        | 24576/150370 [00:03<00:17, 7189.08it/s]","\rparsing log, completed traces ::  17%|#6        | 25310/150370 [00:04<00:33, 3700.97it/s]","\rparsing log, completed traces ::  17%|#7        | 25938/150370 [00:04<00:30, 4135.11it/s]","\rparsing log, completed traces ::  18%|#7        | 26539/150370 [00:04<00:27, 4499.76it/s]","\rparsing log, completed traces ::  18%|#8        | 27149/150370 [00:04<00:25, 4846.64it/s]","\rparsing log, completed traces ::  18%|#8        | 27757/150370 [00:04<00:23, 5137.60it/s]","\rparsing log, completed traces ::  19%|#8        | 28427/150370 [00:04<00:22, 5532.00it/s]","\rparsing log, completed traces ::  19%|#9        | 29175/150370 [00:04<00:20, 6032.69it/s]","\rparsing log, completed traces ::  20%|#9        | 29885/150370 [00:04<00:19, 6323.83it/s]","\rparsing log, completed traces ::  20%|##        | 30557/150370 [00:04<00:18, 6338.15it/s]","\rparsing log, completed traces ::  21%|##        | 31219/150370 [00:05<00:18, 6333.67it/s]","\rparsing log, completed traces ::  21%|##1       | 31872/150370 [00:05<00:18, 6320.55it/s]","\rparsing log, completed traces ::  22%|##1       | 32518/150370 [00:05<00:18, 6334.26it/s]","\rparsing log, completed traces ::  22%|##2       | 33161/150370 [00:05<00:18, 6340.34it/s]","\rparsing log, completed traces ::  22%|##2       | 33813/150370 [00:05<00:18, 6390.63it/s]","\rparsing log, completed traces ::  23%|##2       | 34457/150370 [00:05<00:18, 6341.57it/s]","\rparsing log, completed traces ::  23%|##3       | 35095/150370 [00:06<00:37, 3038.81it/s]","\rparsing log, completed traces ::  24%|##3       | 35733/150370 [00:06<00:31, 3597.51it/s]","\rparsing log, completed traces ::  24%|##4       | 36375/150370 [00:06<00:27, 4133.96it/s]","\rparsing log, completed traces ::  25%|##4       | 37023/150370 [00:06<00:24, 4636.07it/s]","\rparsing log, completed traces ::  25%|##5       | 37666/150370 [00:06<00:22, 5057.54it/s]","\rparsing log, completed traces ::  25%|##5       | 38330/150370 [00:06<00:20, 5455.37it/s]","\rparsing log, completed traces ::  26%|##5       | 38999/150370 [00:06<00:19, 5781.11it/s]","\rparsing log, completed traces ::  26%|##6       | 39687/150370 [00:06<00:18, 6081.60it/s]","\rparsing log, completed traces ::  27%|##6       | 40349/150370 [00:06<00:17, 6232.41it/s]","\rparsing log, completed traces ::  27%|##7       | 41021/150370 [00:06<00:17, 6370.67it/s]","\rparsing log, completed traces ::  28%|##7       | 41725/150370 [00:07<00:16, 6561.62it/s]","\rparsing log, completed traces ::  28%|##8       | 42441/150370 [00:07<00:16, 6734.30it/s]","\rparsing log, completed traces ::  29%|##8       | 43136/150370 [00:07<00:15, 6797.67it/s]","\rparsing log, completed traces ::  29%|##9       | 43833/150370 [00:07<00:15, 6845.92it/s]","\rparsing log, completed traces ::  30%|##9       | 44542/150370 [00:07<00:15, 6915.36it/s]","\rparsing log, completed traces ::  30%|###       | 45238/150370 [00:07<00:15, 6871.85it/s]","\rparsing log, completed traces ::  31%|###       | 45929/150370 [00:07<00:15, 6860.68it/s]","\rparsing log, completed traces ::  31%|###1      | 46618/150370 [00:07<00:15, 6815.29it/s]","\rparsing log, completed traces ::  31%|###1      | 47302/150370 [00:07<00:15, 6740.39it/s]","\rparsing log, completed traces ::  32%|###1      | 47978/150370 [00:08<00:32, 3140.93it/s]","\rparsing log, completed traces ::  32%|###2      | 48722/150370 [00:08<00:26, 3844.15it/s]","\rparsing log, completed traces ::  33%|###2      | 49491/150370 [00:08<00:22, 4564.61it/s]","\rparsing log, completed traces ::  33%|###3      | 50197/150370 [00:08<00:19, 5095.66it/s]","\rparsing log, completed traces ::  34%|###3      | 50882/150370 [00:08<00:18, 5492.55it/s]","\rparsing log, completed traces ::  34%|###4      | 51586/150370 [00:08<00:16, 5878.07it/s]","\rparsing log, completed traces ::  35%|###4      | 52279/150370 [00:08<00:15, 6153.79it/s]","\rparsing log, completed traces ::  35%|###5      | 52980/150370 [00:09<00:15, 6380.81it/s]","\rparsing log, completed traces ::  36%|###5      | 53666/150370 [00:09<00:14, 6490.33it/s]","\rparsing log, completed traces ::  36%|###6      | 54428/150370 [00:09<00:14, 6811.50it/s]","\rparsing log, completed traces ::  37%|###6      | 55167/150370 [00:09<00:13, 6976.86it/s]","\rparsing log, completed traces ::  37%|###7      | 55884/150370 [00:09<00:13, 6992.09it/s]","\rparsing log, completed traces ::  38%|###7      | 56634/150370 [00:09<00:13, 7140.39it/s]","\rparsing log, completed traces ::  38%|###8      | 57367/150370 [00:09<00:12, 7190.73it/s]","\rparsing log, completed traces ::  39%|###8      | 58093/150370 [00:09<00:12, 7153.83it/s]","\rparsing log, completed traces ::  39%|###9      | 58814/150370 [00:09<00:12, 7065.87it/s]","\rparsing log, completed traces ::  40%|###9      | 59536/150370 [00:09<00:12, 7111.11it/s]","\rparsing log, completed traces ::  40%|####      | 60251/150370 [00:10<00:12, 7122.16it/s]","\rparsing log, completed traces ::  41%|####      | 61030/150370 [00:10<00:12, 7319.41it/s]","\rparsing log, completed traces ::  41%|####1     | 61829/150370 [00:10<00:11, 7503.82it/s]","\rparsing log, completed traces ::  42%|####1     | 62594/150370 [00:10<00:11, 7522.08it/s]","\rparsing log, completed traces ::  42%|####2     | 63347/150370 [00:10<00:27, 3210.04it/s]","\rparsing log, completed traces ::  43%|####2     | 64076/150370 [00:11<00:22, 3835.97it/s]","\rparsing log, completed traces ::  43%|####3     | 64812/150370 [00:11<00:19, 4469.70it/s]","\rparsing log, completed traces ::  44%|####3     | 65549/150370 [00:11<00:16, 5062.64it/s]","\rparsing log, completed traces ::  44%|####4     | 66295/150370 [00:11<00:15, 5604.35it/s]","\rparsing log, completed traces ::  45%|####4     | 67047/150370 [00:11<00:13, 6068.54it/s]","\rparsing log, completed traces ::  45%|####5     | 67894/150370 [00:11<00:12, 6689.01it/s]","\rparsing log, completed traces ::  46%|####5     | 68648/150370 [00:11<00:11, 6909.86it/s]","\rparsing log, completed traces ::  46%|####6     | 69401/150370 [00:11<00:11, 7076.37it/s]","\rparsing log, completed traces ::  47%|####6     | 70154/150370 [00:11<00:11, 7136.21it/s]","\rparsing log, completed traces ::  47%|####7     | 70899/150370 [00:11<00:11, 7216.86it/s]","\rparsing log, completed traces ::  48%|####7     | 71643/150370 [00:12<00:10, 7228.01it/s]","\rparsing log, completed traces ::  48%|####8     | 72382/150370 [00:12<00:10, 7240.20it/s]","\rparsing log, completed traces ::  49%|####8     | 73117/150370 [00:12<00:10, 7120.38it/s]","\rparsing log, completed traces ::  49%|####9     | 73838/150370 [00:12<00:10, 6966.37it/s]","\rparsing log, completed traces ::  50%|####9     | 74679/150370 [00:12<00:10, 7363.26it/s]","\rparsing log, completed traces ::  50%|#####     | 75435/150370 [00:12<00:10, 7419.63it/s]","\rparsing log, completed traces ::  51%|#####     | 76182/150370 [00:12<00:10, 7363.58it/s]","\rparsing log, completed traces ::  51%|#####1    | 76927/150370 [00:12<00:09, 7382.53it/s]","\rparsing log, completed traces ::  52%|#####1    | 77668/150370 [00:12<00:10, 7263.26it/s]","\rparsing log, completed traces ::  52%|#####2    | 78397/150370 [00:12<00:09, 7235.72it/s]","\rparsing log, completed traces ::  53%|#####2    | 79122/150370 [00:13<00:09, 7232.91it/s]","\rparsing log, completed traces ::  53%|#####3    | 79847/150370 [00:13<00:09, 7213.51it/s]","\rparsing log, completed traces ::  54%|#####3    | 80569/150370 [00:13<00:09, 7191.90it/s]","\rparsing log, completed traces ::  54%|#####4    | 81289/150370 [00:13<00:09, 7169.31it/s]","\rparsing log, completed traces ::  55%|#####4    | 82007/150370 [00:14<00:25, 2727.68it/s]","\rparsing log, completed traces ::  55%|#####5    | 82801/150370 [00:14<00:19, 3451.67it/s]","\rparsing log, completed traces ::  56%|#####5    | 83551/150370 [00:14<00:16, 4121.70it/s]","\rparsing log, completed traces ::  56%|#####6    | 84279/150370 [00:14<00:14, 4716.61it/s]","\rparsing log, completed traces ::  57%|#####6    | 85014/150370 [00:14<00:12, 5280.08it/s]","\rparsing log, completed traces ::  57%|#####7    | 85734/150370 [00:14<00:11, 5729.02it/s]","\rparsing log, completed traces ::  58%|#####7    | 86484/150370 [00:14<00:10, 6171.33it/s]","\rparsing log, completed traces ::  58%|#####8    | 87237/150370 [00:14<00:09, 6527.04it/s]","\rparsing log, completed traces ::  59%|#####8    | 87984/150370 [00:14<00:09, 6783.90it/s]","\rparsing log, completed traces ::  59%|#####9    | 88748/150370 [00:14<00:08, 7000.07it/s]","\rparsing log, completed traces ::  60%|#####9    | 89500/150370 [00:15<00:08, 7148.28it/s]","\rparsing log, completed traces ::  60%|######    | 90243/150370 [00:15<00:08, 7219.29it/s]","\rparsing log, completed traces ::  61%|######    | 90991/150370 [00:15<00:08, 7295.02it/s]","\rparsing log, completed traces ::  61%|######1   | 91745/150370 [00:15<00:07, 7365.40it/s]","\rparsing log, completed traces ::  62%|######1   | 92492/150370 [00:15<00:07, 7385.95it/s]","\rparsing log, completed traces ::  62%|######2   | 93244/150370 [00:15<00:07, 7425.01it/s]","\rparsing log, completed traces ::  63%|######2   | 94004/150370 [00:15<00:07, 7475.33it/s]","\rparsing log, completed traces ::  63%|######3   | 94797/150370 [00:15<00:07, 7589.67it/s]","\rparsing log, completed traces ::  64%|######3   | 95565/150370 [00:15<00:07, 7615.94it/s]","\rparsing log, completed traces ::  64%|######4   | 96329/150370 [00:15<00:07, 7580.33it/s]","\rparsing log, completed traces ::  65%|######4   | 97089/150370 [00:16<00:07, 7505.06it/s]","\rparsing log, completed traces ::  65%|######5   | 97841/150370 [00:16<00:07, 7467.68it/s]","\rparsing log, completed traces ::  66%|######5   | 98589/150370 [00:16<00:06, 7461.43it/s]","\rparsing log, completed traces ::  66%|######6   | 99336/150370 [00:16<00:06, 7453.83it/s]","\rparsing log, completed traces ::  67%|######6   | 100082/150370 [00:16<00:06, 7377.96it/s]","\rparsing log, completed traces ::  67%|######7   | 100831/150370 [00:16<00:06, 7408.60it/s]","\rparsing log, completed traces ::  68%|######7   | 101584/150370 [00:16<00:06, 7444.30it/s]","\rparsing log, completed traces ::  68%|######8   | 102329/150370 [00:16<00:06, 7434.70it/s]","\rparsing log, completed traces ::  69%|######8   | 103073/150370 [00:17<00:17, 2664.61it/s]","\rparsing log, completed traces ::  69%|######9   | 103796/150370 [00:17<00:14, 3269.33it/s]","\rparsing log, completed traces ::  70%|######9   | 104509/150370 [00:17<00:11, 3883.57it/s]","\rparsing log, completed traces ::  70%|######9   | 105223/150370 [00:17<00:10, 4484.13it/s]","\rparsing log, completed traces ::  70%|#######   | 105940/150370 [00:17<00:08, 5042.50it/s]","\rparsing log, completed traces ::  71%|#######   | 106666/150370 [00:17<00:07, 5537.22it/s]","\rparsing log, completed traces ::  71%|#######1  | 107395/150370 [00:18<00:07, 5968.42it/s]","\rparsing log, completed traces ::  72%|#######1  | 108106/150370 [00:18<00:06, 6265.42it/s]","\rparsing log, completed traces ::  72%|#######2  | 108827/150370 [00:18<00:06, 6504.02it/s]","\rparsing log, completed traces ::  73%|#######2  | 109544/150370 [00:18<00:06, 6687.91it/s]","\rparsing log, completed traces ::  73%|#######3  | 110264/150370 [00:18<00:05, 6826.74it/s]","\rparsing log, completed traces ::  74%|#######3  | 110995/150370 [00:18<00:05, 6966.10it/s]","\rparsing log, completed traces ::  74%|#######4  | 111719/150370 [00:18<00:05, 7045.04it/s]","\rparsing log, completed traces ::  75%|#######4  | 112439/150370 [00:18<00:05, 7063.86it/s]","\rparsing log, completed traces ::  75%|#######5  | 113156/150370 [00:18<00:05, 7078.86it/s]","\rparsing log, completed traces ::  76%|#######5  | 113948/150370 [00:18<00:04, 7326.11it/s]","\rparsing log, completed traces ::  76%|#######6  | 114853/150370 [00:19<00:04, 7818.47it/s]","\rparsing log, completed traces ::  77%|#######6  | 115755/150370 [00:19<00:04, 8155.80it/s]","\rparsing log, completed traces ::  78%|#######7  | 116662/150370 [00:19<00:04, 8400.47it/s]","\rparsing log, completed traces ::  78%|#######8  | 117577/150370 [00:19<00:03, 8623.11it/s]","\rparsing log, completed traces ::  79%|#######8  | 118441/150370 [00:19<00:03, 8010.35it/s]","\rparsing log, completed traces ::  79%|#######9  | 119252/150370 [00:19<00:04, 7604.78it/s]","\rparsing log, completed traces ::  80%|#######9  | 120023/150370 [00:19<00:04, 7345.35it/s]","\rparsing log, completed traces ::  80%|########  | 120765/150370 [00:19<00:04, 7163.97it/s]","\rparsing log, completed traces ::  81%|########  | 121487/150370 [00:19<00:04, 7039.34it/s]","\rparsing log, completed traces ::  81%|########1 | 122194/150370 [00:20<00:04, 6985.65it/s]","\rparsing log, completed traces ::  82%|########1 | 122895/150370 [00:20<00:03, 6991.57it/s]","\rparsing log, completed traces ::  82%|########2 | 123596/150370 [00:20<00:03, 6903.50it/s]","\rparsing log, completed traces ::  83%|########2 | 124288/150370 [00:20<00:03, 6823.09it/s]","\rparsing log, completed traces ::  83%|########3 | 124971/150370 [00:20<00:03, 6816.56it/s]","\rparsing log, completed traces ::  84%|########3 | 125666/150370 [00:20<00:03, 6837.62it/s]","\rparsing log, completed traces ::  84%|########4 | 126352/150370 [00:20<00:03, 6842.09it/s]","\rparsing log, completed traces ::  84%|########4 | 127037/150370 [00:21<00:10, 2256.37it/s]","\rparsing log, completed traces ::  85%|########4 | 127718/150370 [00:21<00:08, 2809.57it/s]","\rparsing log, completed traces ::  85%|########5 | 128408/150370 [00:21<00:06, 3416.73it/s]","\rparsing log, completed traces ::  86%|########5 | 129104/150370 [00:21<00:05, 4036.27it/s]","\rparsing log, completed traces ::  86%|########6 | 129796/150370 [00:21<00:04, 4612.98it/s]","\rparsing log, completed traces ::  87%|########6 | 130494/150370 [00:21<00:03, 5137.01it/s]","\rparsing log, completed traces ::  87%|########7 | 131186/150370 [00:22<00:03, 5566.39it/s]","\rparsing log, completed traces ::  88%|########7 | 131949/150370 [00:22<00:03, 6093.12it/s]","\rparsing log, completed traces ::  88%|########8 | 132762/150370 [00:22<00:02, 6616.98it/s]","\rparsing log, completed traces ::  89%|########8 | 133534/150370 [00:22<00:02, 6902.81it/s]","\rparsing log, completed traces ::  89%|########9 | 134274/150370 [00:22<00:02, 6968.16it/s]","\rparsing log, completed traces ::  90%|########9 | 135006/150370 [00:22<00:02, 6991.86it/s]","\rparsing log, completed traces ::  90%|######### | 135730/150370 [00:22<00:02, 6946.08it/s]","\rparsing log, completed traces ::  91%|######### | 136442/150370 [00:22<00:01, 6972.01it/s]","\rparsing log, completed traces ::  91%|#########1| 137152/150370 [00:22<00:01, 6932.24it/s]","\rparsing log, completed traces ::  92%|#########1| 137854/150370 [00:22<00:01, 6875.00it/s]","\rparsing log, completed traces ::  92%|#########2| 138552/150370 [00:23<00:01, 6904.96it/s]","\rparsing log, completed traces ::  93%|#########2| 139251/150370 [00:23<00:01, 6929.90it/s]","\rparsing log, completed traces ::  93%|#########3| 139948/150370 [00:23<00:01, 6931.00it/s]","\rparsing log, completed traces ::  94%|#########3| 140700/150370 [00:23<00:01, 7103.28it/s]","\rparsing log, completed traces ::  94%|#########4| 141412/150370 [00:23<00:01, 7084.73it/s]","\rparsing log, completed traces ::  95%|#########4| 142123/150370 [00:23<00:01, 7091.20it/s]","\rparsing log, completed traces ::  95%|#########4| 142833/150370 [00:23<00:01, 7079.67it/s]","\rparsing log, completed traces ::  95%|#########5| 143553/150370 [00:23<00:00, 7099.08it/s]","\rparsing log, completed traces ::  96%|#########5| 144282/150370 [00:23<00:00, 7140.20it/s]","\rparsing log, completed traces ::  96%|#########6| 144997/150370 [00:23<00:00, 7024.43it/s]","\rparsing log, completed traces ::  97%|#########6| 145701/150370 [00:24<00:00, 6703.45it/s]","\rparsing log, completed traces ::  97%|#########7| 146375/150370 [00:24<00:00, 6342.11it/s]","\rparsing log, completed traces ::  98%|#########7| 147015/150370 [00:24<00:00, 6174.87it/s]","\rparsing log, completed traces ::  98%|#########8| 147637/150370 [00:24<00:00, 6066.19it/s]","\rparsing log, completed traces ::  99%|#########8| 148246/150370 [00:24<00:00, 5974.04it/s]","\rparsing log, completed traces ::  99%|#########8| 148855/150370 [00:24<00:00, 6004.81it/s]","\rparsing log, completed traces ::  99%|#########9| 149488/150370 [00:24<00:00, 6096.06it/s]","\rparsing log, completed traces :: 100%|#########9| 150170/150370 [00:24<00:00, 6293.71it/s]","","\rparsing log, completed traces :: 100%|##########| 150370/150370 [00:24<00:00, 6052.13it/s]","\n","[data] Loaded datasets: ['BPI2012', 'BPI2017', 'ROAD']","\n","\n=== Dataset: BPI2012 ===","\n","Traceback (most recent call last):\n  File \"/workspace/.venv/lib/python3.10/site-packages/pandas/core/indexes/base.py\", line 3812, in get_loc\n    return self._engine.get_loc(casted_key)\n  File \"pandas/_libs/index.pyx\", line 167, in pandas._libs.index.IndexEngine.get_loc\n  File \"pandas/_libs/index.pyx\", line 196, in pandas._libs.index.IndexEngine.get_loc\n  File \"pandas/_libs/hashtable_class_helper.pxi\", line 2606, in pandas._libs.hashtable.Int64HashTable.get_item\n  File \"pandas/_libs/hashtable_class_helper.pxi\", line 2630, in pandas._libs.hashtable.Int64HashTable.get_item\nKeyError: 4\n\nThe above exception was the direct cause of the following exception:\n\nTraceback (most recent call last):\n  File \"runfile.py\", line 497, in <module>\n    main()\n  File \"runfile.py\", line 484, in main\n    train_one_dataset(\n  File \"runfile.py\", line 269, in train_one_dataset\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n  File \"runfile.py\", line 116, in build_prefix_dataset\n    \"next_ts\": ts[k],\n  File \"/workspace/.venv/lib/python3.10/site-packages/pandas/core/series.py\", line 1130, in __getitem__\n    return self._get_value(key)\n  File \"/workspace/.venv/lib/python3.10/site-packages/pandas/core/series.py\", line 1246, in _get_value\n    loc = self.index.get_loc(label)\n  File \"/workspace/.venv/lib/python3.10/site-packages/pandas/core/indexes/base.py\", line 3819, in get_loc\n    raise KeyError(key) from err\nKeyError: 4\n","Execution time: 2 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"","parse_metrics_code":"","parse_term_out":null,"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":149.92952132225037,"exc_type":"KeyError","exc_info":{"args":["4"]},"exc_stack":[["/workspace/ai_scientist/treesearch/interpreter.py",144,"_run_session","exec(compile(code, self.agent_file_name, \"exec\"), global_scope)"],["runfile.py",497,"<module>","main()"],["runfile.py",484,"main","train_one_dataset("],["runfile.py",269,"train_one_dataset","samples_all, act2id, id2act, pad_id = build_prefix_dataset("],["runfile.py",116,"build_prefix_dataset","\"next_ts\": ts[k],"],["/workspace/.venv/lib/python3.10/site-packages/pandas/core/series.py",1130,"__getitem__","return self._get_value(key)"],["/workspace/.venv/lib/python3.10/site-packages/pandas/core/series.py",1246,"_get_value","loc = self.index.get_loc(label)"],["/workspace/.venv/lib/python3.10/site-packages/pandas/core/indexes/base.py",3819,"get_loc","raise KeyError(key) from err"]],"analysis":"Execution failed with a KeyError during prefix construction. Root cause: timestamp series ts is a pandas Series with original case group index, but code indexes it positionally (ts[k], ts[k-1]) and also uses ts[0] when computing deltas/since_start. These are label-based lookups in pandas and raise KeyError if the index labels are not 0..T-1. Fix: convert to a NumPy array or use .to_numpy()/.values and positional indexing, or use .iloc for pandas positional access. Example fix in build_prefix_dataset:\n\n    ts_series = pd.to_datetime(g['timestamp'])\n    ts_vals = ts_series.view('int64') // 10**9  # or ts_series.astype('int64').to_numpy()\n    deltas = np.diff(ts_vals, prepend=ts_vals[0])\n    since_start = ts_vals - ts_vals[0]\n    ...\n    'last_ts': ts_vals[k-1],\n    'next_ts': ts_vals[k],\n\nAdditionally, there is a data leakage/normalization bug: build_prefix_dataset already normalizes features using statistics computed over all samples (train+val+test), and later the code re-normalizes using train-only stats. This double-normalizes and leaks information from validation/test into feature scaling. Fix by removing normalization from build_prefix_dataset (return raw features) and performing normalization once using stats computed strictly from train samples, applying them to val/test. Alternatively, add a flag to build_prefix_dataset to return raw features and perform scaling outside.\n\nAfter applying these fixes, the pipeline should run: load XES, time-based split by case start, build prefixes, train LSTM baseline, and compute next-activity metrics (accuracy, macro-F1, top-3).","exp_results_dir":null,"metric":{"value":null,"maximize":null,"name":null,"description":null},"is_buggy":true,"is_buggy_plots":null,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":[],"plot_paths":[],"plot_analyses":[],"vlm_feedback_summary":[],"datasets_successfully_tested":[],"ablation_name":null,"hyperparam_name":null,"is_seed_node":false,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"import os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom pathlib import Path\nfrom collections import defaultdict, Counter\nfrom datetime import datetime, timezone\nfrom sklearn.metrics import f1_score, accuracy_score\nimport matplotlib.pyplot as plt\n\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# --------------------- Experiment data container ---------------------\nexperiment_data = {\n    \"PPM_NextActivity\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},  # store dicts per epoch/split\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],  # test predictions (labels)\n        \"probs\": [],  # test probabilities\n        \"ground_truth\": [],  # test labels\n        \"epochs\": [],\n        \"feature_norm\": {},\n    }\n}\n\n\n# --------------------- Data discovery and XES loading ---------------------\ndef _has_xes(dirpath: Path) -> bool:\n    try:\n        return dirpath.is_dir() and (\n            any(dirpath.glob(\"*.xes\")) or any(dirpath.glob(\"*.xes.gz\"))\n        )\n    except Exception:\n        return False\n\n\ndef _resolve_data_dir() -> Path:\n    candidates = []\n    candidates += [Path(\"input\").resolve(), (Path.cwd() / \"input\").resolve()]\n    cwd = Path.cwd().resolve()\n    for base in [cwd, *cwd.parents]:\n        candidates.append((base / \"data\").resolve())\n        candidates.append((base / \"input\").resolve())\n    candidates += [\n        Path(\"/workspace/input\"),\n        Path(\"/workspace/data\"),\n        Path(\"/workspace/ai_scientist/data\"),\n        Path(\"/workspace/AI-Scientist-v2/data\"),\n        Path(\"/workspace/experiments/data\"),\n        Path(\"/workspace/ai_scientist/input\"),\n        Path(\"/workspace/experiments/input\"),\n    ]\n    seen = set()\n    for p in candidates:\n        if p in seen:\n            continue\n        seen.add(p)\n        if _has_xes(p):\n            print(f\"[data] Using discovered data dir: {p}\")\n            return p\n    raise FileNotFoundError(\n        \"Could not locate a directory containing .xes or .xes.gz files under input/ or data/.\"\n    )\n\n\ndef _first_match(d: Path, patterns):\n    for pat in patterns:\n        for p in d.glob(pat):\n            if p.is_file():\n                return p\n    return None\n\n\ndef load_known_dataset():\n    data_dir = _resolve_data_dir()\n    patterns = {\n        \"BPI2017\": [\"BPI_Challenge_2017*.xes*\", \"BPI2017*.xes*\", \"*2017*.xes*\"],\n        \"BPI2012\": [\"BPI_Challenge_2012*.xes*\", \"BPI2012*.xes*\", \"*2012*.xes*\"],\n        \"ROAD\": [\n            \"Road_Traffic_Fine_Management_Process*.xes*\",\n            \"*Traffic*Fine*.xes*\",\n            \"*Traffic*.xes*\",\n        ],\n    }\n    order = (\"BPI2012\", \"BPI2017\", \"ROAD\")  # prefer BPI2012 per instruction\n    try:\n        from pm4py.objects.log.importer.xes import importer as xes_importer\n    except Exception as e:\n        raise ImportError(\"pm4py is required. Please install pm4py.\") from e\n    chosen_name, path = None, None\n    for name in order:\n        p = _first_match(data_dir, patterns[name])\n        if p is not None:\n            chosen_name, path = name, p\n            break\n    if path is None:\n        # fallback to any xes\n        files = list(data_dir.glob(\"*.xes\")) + list(data_dir.glob(\"*.xes.gz\"))\n        if not files:\n            raise FileNotFoundError(\"No XES files found.\")\n        chosen_name, path = files[0].stem, files[0]\n    print(f\"[data] Loading dataset {chosen_name} from {path}\")\n    log = xes_importer.apply(str(path))\n\n    rows = []\n    for tr in log:\n        case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n            \"case:concept:name\"\n        )\n        for e in tr:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": e.get(\"concept:name\"),\n                    \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                    \"timestamp\": e.get(\"time:timestamp\"),\n                    \"resource\": e.get(\"org:resource\", \"System\"),\n                }\n            )\n    df = pd.DataFrame(rows)\n    df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n    df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n    # keep only lifecycle=complete if present\n    if \"lifecycle\" in df.columns and df[\"lifecycle\"].notna().any():\n        complete_mask = df[\"lifecycle\"].str.lower().eq(\"complete\")\n        if complete_mask.sum() > 0:\n            df = df[complete_mask]\n    df = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n    return chosen_name, df\n\n\n# --------------------- Prefix creation and features ---------------------\ndef build_prefix_samples(df, max_cases=4000, max_k=10, min_k=1):\n    # case start times for splitting\n    case_groups = df.groupby(\"case_id\")\n    # Limit number of cases by earliest start time (time-based selection to keep order)\n    case_starts = case_groups[\"timestamp\"].min().sort_values()\n    selected_cases = case_starts.index[:max_cases]\n    df = df[df[\"case_id\"].isin(selected_cases)].copy()\n\n    # Compute case start per event\n    case_start = df.groupby(\"case_id\")[\"timestamp\"].transform(\"min\")\n    df[\"elapsed_since_start\"] = (df[\"timestamp\"] - case_start).dt.total_seconds()\n    df[\"hour\"] = df[\"timestamp\"].dt.hour.astype(int)\n    df[\"weekday\"] = df[\"timestamp\"].dt.weekday.astype(int)\n    df[\"is_working_time\"] = (\n        (df[\"weekday\"] < 5) & (df[\"hour\"] >= 9) & (df[\"hour\"] <= 17)\n    ).astype(int)\n\n    # Build sequences\n    sequences = []\n    for cid, g in df.groupby(\"case_id\"):\n        acts = g[\"activity\"].tolist()\n        times = g[\"timestamp\"].tolist()\n        elapsed = g[\"elapsed_since_start\"].tolist()\n        hours = g[\"hour\"].tolist()\n        wdays = g[\"weekday\"].tolist()\n        workf = g[\"is_working_time\"].tolist()\n        if len(acts) < min_k + 1:\n            continue\n        # inter-arrival times\n        delta = [0.0]\n        for i in range(1, len(times)):\n            delta.append((times[i] - times[i - 1]).total_seconds())\n        for k in range(min_k, min(max_k, len(acts) - 1) + 1):\n            prefix_acts = acts[:k]\n            # per-step numeric features aligned with prefix\n            x_elapsed = elapsed[:k]\n            x_delta = delta[:k]\n            x_hour = hours[:k]\n            x_wday = wdays[:k]\n            x_work = workf[:k]\n            target = acts[k]  # next activity\n            sequences.append(\n                {\n                    \"case_id\": cid,\n                    \"k\": k,\n                    \"prefix_acts\": prefix_acts,\n                    \"feat\": np.stack(\n                        [x_elapsed, x_delta, x_hour, x_wday, x_work], axis=1\n                    ),  # shape (k, 5)\n                    \"target\": target,\n                    \"case_start\": times[0],\n                }\n            )\n    return sequences\n\n\ndef time_based_split(sequences, train_ratio=0.7, val_ratio=0.1):\n    # Split by case start time (earlier cases -> train)\n    df_seq = pd.DataFrame(\n        [{\"case_id\": s[\"case_id\"], \"case_start\": s[\"case_start\"]} for s in sequences]\n    ).drop_duplicates()\n    df_seq = df_seq.sort_values(\"case_start\")\n    n = len(df_seq)\n    n_train = int(n * train_ratio)\n    n_val = int(n * val_ratio)\n    train_cases = set(df_seq[\"case_id\"].iloc[:n_train])\n    val_cases = set(df_seq[\"case_id\"].iloc[n_train : n_train + n_val])\n    test_cases = set(df_seq[\"case_id\"].iloc[n_train + n_val :])\n    train = [s for s in sequences if s[\"case_id\"] in train_cases]\n    val = [s for s in sequences if s[\"case_id\"] in val_cases]\n    test = [s for s in sequences if s[\"case_id\"] in test_cases]\n    return train, val, test\n\n\n# --------------------- Vocabulary and encoding ---------------------\nclass Vocab:\n    def __init__(self, tokens, min_freq=1, add_unk=True, add_pad=True):\n        counter = Counter(tokens)\n        self.itos = []\n        if add_pad:\n            self.itos.append(\"<PAD>\")\n        if add_unk:\n            self.itos.append(\"<UNK>\")\n        for t, c in counter.items():\n            if c >= min_freq:\n                self.itos.append(t)\n        self.stoi = {t: i for i, t in enumerate(self.itos)}\n        self.pad_idx = self.stoi.get(\"<PAD>\", 0)\n        self.unk_idx = self.stoi.get(\"<UNK>\", 1 if \"<UNK>\" in self.stoi else 0)\n\n    def encode(self, seq):\n        return [self.stoi.get(t, self.unk_idx) for t in seq]\n\n    def __len__(self):\n        return len(self.itos)\n\n\n# --------------------- Dataset and collate ---------------------\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, act_vocab, max_len):\n        self.samples = samples\n        self.act_vocab = act_vocab\n        self.max_len = max_len\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        x_ids = torch.tensor(self.act_vocab.encode(s[\"prefix_acts\"]), dtype=torch.long)\n        x_num = torch.tensor(s[\"feat\"], dtype=torch.float32)\n        y = torch.tensor(\n            self.act_vocab.stoi.get(s[\"target\"], self.act_vocab.unk_idx),\n            dtype=torch.long,\n        )\n        length = torch.tensor(len(s[\"prefix_acts\"]), dtype=torch.long)\n        return {\"x_ids\": x_ids, \"x_num\": x_num, \"y\": y, \"len\": length}\n\n\ndef collate_fn(batch, pad_idx, max_len=None):\n    B = len(batch)\n    lengths = torch.stack([b[\"len\"] for b in batch])\n    L = int(lengths.max().item()) if max_len is None else max_len\n    x_ids = torch.full((B, L), pad_idx, dtype=torch.long)\n    x_num = torch.zeros((B, L, batch[0][\"x_num\"].shape[1]), dtype=torch.float32)\n    for i, b in enumerate(batch):\n        l = min(len(b[\"x_ids\"]), L)\n        x_ids[i, :l] = b[\"x_ids\"][:l]\n        x_num[i, :l, :] = b[\"x_num\"][:l, :]\n    y = torch.stack([b[\"y\"] for b in batch])\n    lengths = torch.clamp(lengths, max=L)\n    return {\"x_ids\": x_ids, \"x_num\": x_num, \"y\": y, \"len\": lengths}\n\n\n# --------------------- Model ---------------------\nclass LSTMNextActivity(nn.Module):\n    def __init__(\n        self, vocab_size, emb_dim, num_feat, hid_dim, num_classes, pad_idx, dropout=0.1\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + num_feat, hidden_size=hid_dim, batch_first=True\n        )\n        self.dropout = nn.Dropout(dropout)\n        self.fc = nn.Linear(hid_dim, num_classes)\n\n    def forward(self, x_ids, x_num, lengths):\n        emb = self.emb(x_ids)  # (B, L, E)\n        x = torch.cat([emb, x_num], dim=-1)\n        # We can use simple last-step based on lengths for simplicity\n        out, (hn, cn) = self.lstm(x)  # (B, L, H), (1,B,H)\n        h_last = hn[-1]  # (B, H)\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)  # (B, C)\n        return logits\n\n\n# --------------------- Metrics ---------------------\ndef topk_acc(probs, y_true, k=3):\n    topk = np.argpartition(-probs, kth=min(k, probs.shape[1] - 1), axis=1)[:, :k]\n    y_true = y_true.reshape(-1, 1)\n    return np.mean((topk == y_true).any(axis=1))\n\n\ndef compute_metrics(logits, y):\n    probs = torch.softmax(logits, dim=1).detach().cpu().numpy()\n    preds = probs.argmax(axis=1)\n    y_true = y.detach().cpu().numpy()\n    acc = accuracy_score(y_true, preds)\n    mf1 = f1_score(y_true, preds, average=\"macro\")\n    t3 = topk_acc(probs, y_true, k=3)\n    return {\"accuracy\": acc, \"macro_f1\": mf1, \"top3\": t3}, preds, probs\n\n\n# --------------------- Training and evaluation loops ---------------------\ndef run_epoch(model, loader, optimizer=None, criterion=None, normalize=None):\n    is_train = optimizer is not None\n    model.train() if is_train else model.eval()\n    total_loss = 0.0\n    all_logits = []\n    all_y = []\n    for batch in loader:\n        # move to device\n        batch = {\n            k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n            for k, v in batch.items()\n        }\n        # normalize numeric features\n        x_num = batch[\"x_num\"]\n        if normalize is not None:\n            mean, std = normalize\n            x_num = (x_num - mean) / (std + 1e-6)\n        if is_train:\n            optimizer.zero_grad()\n        logits = model(batch[\"x_ids\"], x_num, batch[\"len\"])\n        loss = criterion(logits, batch[\"y\"])\n        if is_train:\n            loss.backward()\n            nn.utils.clip_grad_norm_(model.parameters(), 1.0)\n            optimizer.step()\n        total_loss += loss.item() * batch[\"y\"].size(0)\n        all_logits.append(logits.detach().cpu())\n        all_y.append(batch[\"y\"].detach().cpu())\n    if len(all_logits) == 0:\n        return None, None, None\n    all_logits = torch.cat(all_logits, dim=0)\n    all_y = torch.cat(all_y, dim=0)\n    avg_loss = total_loss / all_y.size(0)\n    metrics, preds, probs = compute_metrics(all_logits, all_y)\n    return avg_loss, metrics, (preds, probs, all_y.numpy())\n\n\n# --------------------- Plotting helpers ---------------------\ndef plot_curves(epochs, train_vals, val_vals, ylabel, title, filename):\n    plt.figure(figsize=(6, 4))\n    plt.plot(epochs, train_vals, label=\"train\")\n    plt.plot(epochs, val_vals, label=\"val\")\n    plt.xlabel(\"epoch\")\n    plt.ylabel(ylabel)\n    plt.title(title)\n    plt.legend()\n    plt.tight_layout()\n    plt.savefig(os.path.join(working_dir, filename))\n    plt.close()\n\n\n# --------------------- Pipeline ---------------------\n# Load data\ndataset_name, df = load_known_dataset()\nprint(f\"[data] Dataset {dataset_name} shape={df.shape}\")\n\n# Build samples\nmax_cases = 4000  # keep runtime reasonable\nmax_k = 10\nsamples = build_prefix_samples(df, max_cases=max_cases, max_k=max_k, min_k=1)\nprint(\n    f\"[data] Built {len(samples)} prefix samples from {len(set([s['case_id'] for s in samples]))} cases.\"\n)\n\n# Split time-based\ntrain_s, val_s, test_s = time_based_split(samples, train_ratio=0.7, val_ratio=0.1)\nprint(f\"[split] Train={len(train_s)} Val={len(val_s)} Test={len(test_s)}\")\n\n# Build vocab on train only\nall_train_tokens = [t for s in train_s for t in s[\"prefix_acts\"] + [s[\"target\"]]]\nact_vocab = Vocab(all_train_tokens, min_freq=1, add_unk=True, add_pad=True)\nnum_classes = len(act_vocab)\nprint(f\"[vocab] Activities: {num_classes}\")\n\n# Prepare datasets and loaders\nbatch_size = 128\ntrain_ds = PrefixDataset(train_s, act_vocab, max_len=max_k)\nval_ds = PrefixDataset(val_s, act_vocab, max_len=max_k)\ntest_ds = PrefixDataset(test_s, act_vocab, max_len=max_k)\ncollate = lambda b: collate_fn(b, pad_idx=act_vocab.pad_idx, max_len=max_k)\ntrain_loader = DataLoader(\n    train_ds, batch_size=batch_size, shuffle=True, collate_fn=collate\n)\nval_loader = DataLoader(\n    val_ds, batch_size=batch_size, shuffle=False, collate_fn=collate\n)\ntest_loader = DataLoader(\n    test_ds, batch_size=batch_size, shuffle=False, collate_fn=collate\n)\n\n# Compute normalization stats on training numeric features\n# Iterate once over train_loader to get mean/std\nnum_feat_dim = 5\n\n\ndef compute_norm(loader):\n    s = torch.zeros(num_feat_dim, device=device)\n    sq = torch.zeros(num_feat_dim, device=device)\n    n = 0\n    for batch in loader:\n        batch = {\n            k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n            for k, v in batch.items()\n        }\n        x = batch[\"x_num\"]  # (B, L, F)\n        s += x.sum(dim=(0, 1))\n        sq += (x * x).sum(dim=(0, 1))\n        n += x.shape[0] * x.shape[1]\n    mean = s / max(n, 1)\n    var = (sq / max(n, 1)) - mean * mean\n    std = torch.sqrt(torch.clamp(var, min=0.0))\n    # reshape for broadcasting (1,1,F)\n    return mean.view(1, 1, -1), std.view(1, 1, -1)\n\n\nnorm_mean, norm_std = compute_norm(train_loader)\nexperiment_data[\"PPM_NextActivity\"][\"feature_norm\"] = {\n    \"mean\": norm_mean.detach().cpu().numpy(),\n    \"std\": norm_std.detach().cpu().numpy(),\n}\n\n# Model\nemb_dim = 64\nhid_dim = 128\nmodel = LSTMNextActivity(\n    vocab_size=len(act_vocab),\n    emb_dim=emb_dim,\n    num_feat=num_feat_dim,\n    hid_dim=hid_dim,\n    num_classes=num_classes,\n    pad_idx=act_vocab.pad_idx,\n    dropout=0.2,\n).to(device)\ncriterion = nn.CrossEntropyLoss()\noptimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n\n# Training\nepochs = 10\nbest_val_loss = float(\"inf\")\nbest_state = None\nfor epoch in range(1, epochs + 1):\n    train_loss, train_metrics, _ = run_epoch(\n        model,\n        train_loader,\n        optimizer=optimizer,\n        criterion=criterion,\n        normalize=(norm_mean, norm_std),\n    )\n    val_loss, val_metrics, _ = run_epoch(\n        model,\n        val_loader,\n        optimizer=None,\n        criterion=criterion,\n        normalize=(norm_mean, norm_std),\n    )\n    experiment_data[\"PPM_NextActivity\"][\"epochs\"].append(epoch)\n    experiment_data[\"PPM_NextActivity\"][\"losses\"][\"train\"].append(\n        (epoch, float(train_loss if train_loss is not None else np.nan))\n    )\n    experiment_data[\"PPM_NextActivity\"][\"losses\"][\"val\"].append(\n        (epoch, float(val_loss if val_loss is not None else np.nan))\n    )\n    experiment_data[\"PPM_NextActivity\"][\"metrics\"][\"train\"].append(\n        (epoch, train_metrics)\n    )\n    experiment_data[\"PPM_NextActivity\"][\"metrics\"][\"val\"].append((epoch, val_metrics))\n    print(\n        f\"Epoch {epoch}: validation_loss = {val_loss:.4f}\"\n        if val_loss is not None\n        else f\"Epoch {epoch}: validation_loss = nan\"\n    )\n    if val_loss is not None and val_loss < best_val_loss:\n        best_val_loss = val_loss\n        best_state = {k: v.cpu() for k, v in model.state_dict().items()}\n\n# Load best\nif best_state is not None:\n    model.load_state_dict({k: v.to(device) for k, v in best_state.items()})\n\n# Evaluate on train/val/test\ntrain_loss, train_metrics, _ = run_epoch(\n    model,\n    train_loader,\n    optimizer=None,\n    criterion=criterion,\n    normalize=(norm_mean, norm_std),\n)\nval_loss, val_metrics, _ = run_epoch(\n    model,\n    val_loader,\n    optimizer=None,\n    criterion=criterion,\n    normalize=(norm_mean, norm_std),\n)\ntest_loss, test_metrics, (test_preds, test_probs, test_y) = run_epoch(\n    model,\n    test_loader,\n    optimizer=None,\n    criterion=criterion,\n    normalize=(norm_mean, norm_std),\n)\n\nprint(\n    f\"[final] Train: acc={train_metrics['accuracy']:.4f} macroF1={train_metrics['macro_f1']:.4f} top3={train_metrics['top3']:.4f} loss={train_loss:.4f}\"\n)\nprint(\n    f\"[final] Val:   acc={val_metrics['accuracy']:.4f} macroF1={val_metrics['macro_f1']:.4f} top3={val_metrics['top3']:.4f} loss={val_loss:.4f}\"\n)\nprint(\n    f\"[final] Test:  acc={test_metrics['accuracy']:.4f} macroF1={test_metrics['macro_f1']:.4f} top3={test_metrics['top3']:.4f} loss={test_loss:.4f}\"\n)\n\n# Save metrics per split including test\nexperiment_data[\"PPM_NextActivity\"][\"metrics\"][\"train\"].append((\"final\", train_metrics))\nexperiment_data[\"PPM_NextActivity\"][\"metrics\"][\"val\"].append((\"final\", val_metrics))\nexperiment_data[\"PPM_NextActivity\"][\"metrics\"][\"test\"].append((\"final\", test_metrics))\nexperiment_data[\"PPM_NextActivity\"][\"predictions\"] = test_preds\nexperiment_data[\"PPM_NextActivity\"][\"probs\"] = test_probs\nexperiment_data[\"PPM_NextActivity\"][\"ground_truth\"] = test_y\nnp.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n\n# Plot loss and Top-3 over epochs\nepochs_axis = experiment_data[\"PPM_NextActivity\"][\"epochs\"]\ntrain_loss_vals = [v[1] for v in experiment_data[\"PPM_NextActivity\"][\"losses\"][\"train\"]]\nval_loss_vals = [v[1] for v in experiment_data[\"PPM_NextActivity\"][\"losses\"][\"val\"]]\nplot_curves(\n    epochs_axis,\n    train_loss_vals,\n    val_loss_vals,\n    ylabel=\"loss\",\n    title=f\"{dataset_name} LSTM loss\",\n    filename=f\"{dataset_name}_loss.png\",\n)\n\ntrain_top3 = [\n    m[1][\"top3\"] for m in experiment_data[\"PPM_NextActivity\"][\"metrics\"][\"train\"]\n]\nval_top3 = [\n    m[1][\"top3\"]\n    for m in experiment_data[\"PPM_NextActivity\"][\"metrics\"][\"val\"]\n    if isinstance(m[0], int)\n]\nplot_curves(\n    epochs_axis,\n    train_top3,\n    val_top3,\n    ylabel=\"Top-3 Acc\",\n    title=f\"{dataset_name} Top-3 Accuracy\",\n    filename=f\"{dataset_name}_top3.png\",\n)\n\n# Print the required Top-3 metric explicitly\nprint(f\"Top-3 Next-Activity Accuracy (Test): {test_metrics['top3']:.4f}\")\n","plan":"We implement a simple predictive process monitoring baseline for next-activity prediction using local XES logs (BPI 2012 preferred) with a strict time-based split at the case level to avoid leakage. We transform each trace into prefix-target pairs up to max length k, using activities as tokens and appending simple process-aware time features (elapsed since start, since last event, hour, weekday, working-time flag). A 1-layer LSTM with an activity embedding consumes padded sequences plus per-step numeric features and predicts the next activity; we train with cross-entropy and evaluate accuracy, macro-F1, and the required Top-3 Next-Activity Accuracy. We normalize numeric features using training statistics and report metrics on train/val/test; validation loss is tracked and printed each epoch, and all metrics/losses/predictions are saved to working/experiment_data.npy. To keep runtime manageable, we cap the number of cases and limit k. We also generate and save simple plots (loss curves and Top-3 accuracy over epochs). The implementation uses pm4py for robust XES loading and falls back gracefully among the known datasets (BPI 2017, BPI 2012, Road). This provides a clear baseline we can later extend with resource-centric simulation.","overall_plan":"","plot_code":null,"plot_plan":null,"step":1,"id":"d500c8a4b305407c83e6f6c31c2f4556","ctime":1757756017.0724857,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Loading dataset BPI2012 from /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<24:42,  8.83it/s]","\rparsing log, completed traces ::   1%|1         | 145/13087 [00:00<00:16, 795.47it/s]","\rparsing log, completed traces ::   2%|2         | 283/13087 [00:00<00:12, 1049.91it/s]","\rparsing log, completed traces ::   3%|3         | 426/13087 [00:00<00:10, 1191.62it/s]","\rparsing log, completed traces ::   4%|4         | 576/13087 [00:00<00:09, 1300.16it/s]","\rparsing log, completed traces ::   6%|5         | 751/13087 [00:00<00:08, 1450.21it/s]","\rparsing log, completed traces ::   7%|6         | 898/13087 [00:00<00:08, 1443.14it/s]","\rparsing log, completed traces ::   8%|7         | 1044/13087 [00:00<00:08, 1433.24it/s]","\rparsing log, completed traces ::   9%|9         | 1197/13087 [00:00<00:08, 1460.74it/s]","\rparsing log, completed traces ::  10%|#         | 1344/13087 [00:01<00:08, 1458.70it/s]","\rparsing log, completed traces ::  12%|#1        | 1529/13087 [00:01<00:07, 1576.79it/s]","\rparsing log, completed traces ::  13%|#2        | 1687/13087 [00:01<00:13, 863.83it/s] ","\rparsing log, completed traces ::  14%|#4        | 1838/13087 [00:01<00:11, 986.11it/s]","\rparsing log, completed traces ::  15%|#5        | 1979/13087 [00:01<00:10, 1074.99it/s]","\rparsing log, completed traces ::  16%|#6        | 2126/13087 [00:01<00:09, 1165.66it/s]","\rparsing log, completed traces ::  17%|#7        | 2276/13087 [00:01<00:08, 1247.03it/s]","\rparsing log, completed traces ::  19%|#8        | 2437/13087 [00:02<00:07, 1341.35it/s]","\rparsing log, completed traces ::  20%|#9        | 2592/13087 [00:02<00:07, 1398.03it/s]","\rparsing log, completed traces ::  21%|##        | 2742/13087 [00:02<00:07, 1407.11it/s]","\rparsing log, completed traces ::  22%|##2       | 2890/13087 [00:02<00:07, 1394.50it/s]","\rparsing log, completed traces ::  23%|##3       | 3037/13087 [00:02<00:07, 1406.38it/s]","\rparsing log, completed traces ::  24%|##4       | 3181/13087 [00:02<00:07, 1391.16it/s]","\rparsing log, completed traces ::  26%|##6       | 3409/13087 [00:02<00:05, 1642.51it/s]","\rparsing log, completed traces ::  27%|##7       | 3581/13087 [00:02<00:05, 1662.71it/s]","\rparsing log, completed traces ::  29%|##8       | 3753/13087 [00:02<00:05, 1675.72it/s]","\rparsing log, completed traces ::  30%|##9       | 3922/13087 [00:02<00:05, 1628.37it/s]","\rparsing log, completed traces ::  31%|###1      | 4087/13087 [00:03<00:09, 944.91it/s] ","\rparsing log, completed traces ::  33%|###2      | 4309/13087 [00:03<00:07, 1190.16it/s]","\rparsing log, completed traces ::  34%|###4      | 4502/13087 [00:03<00:06, 1349.63it/s]","\rparsing log, completed traces ::  36%|###5      | 4686/13087 [00:03<00:05, 1464.72it/s]","\rparsing log, completed traces ::  37%|###7      | 4859/13087 [00:03<00:05, 1516.20it/s]","\rparsing log, completed traces ::  38%|###8      | 5030/13087 [00:03<00:05, 1560.37it/s]","\rparsing log, completed traces ::  40%|###9      | 5200/13087 [00:03<00:05, 1540.05it/s]","\rparsing log, completed traces ::  41%|####      | 5364/13087 [00:04<00:05, 1518.87it/s]","\rparsing log, completed traces ::  42%|####2     | 5523/13087 [00:04<00:05, 1461.21it/s]","\rparsing log, completed traces ::  43%|####3     | 5678/13087 [00:04<00:04, 1484.72it/s]","\rparsing log, completed traces ::  45%|####4     | 5831/13087 [00:04<00:04, 1459.87it/s]","\rparsing log, completed traces ::  46%|####5     | 6008/13087 [00:04<00:04, 1545.04it/s]","\rparsing log, completed traces ::  47%|####7     | 6165/13087 [00:04<00:04, 1512.79it/s]","\rparsing log, completed traces ::  48%|####8     | 6335/13087 [00:04<00:04, 1565.39it/s]","\rparsing log, completed traces ::  50%|####9     | 6496/13087 [00:04<00:04, 1577.34it/s]","\rparsing log, completed traces ::  51%|#####     | 6655/13087 [00:04<00:04, 1529.94it/s]","\rparsing log, completed traces ::  52%|#####2    | 6809/13087 [00:05<00:07, 810.59it/s] ","\rparsing log, completed traces ::  53%|#####2    | 6929/13087 [00:05<00:07, 878.44it/s]","\rparsing log, completed traces ::  54%|#####4    | 7081/13087 [00:05<00:05, 1006.39it/s]","\rparsing log, completed traces ::  55%|#####5    | 7232/13087 [00:05<00:05, 1117.00it/s]","\rparsing log, completed traces ::  56%|#####6    | 7367/13087 [00:05<00:04, 1163.02it/s]","\rparsing log, completed traces ::  57%|#####7    | 7505/13087 [00:05<00:04, 1217.81it/s]","\rparsing log, completed traces ::  58%|#####8    | 7642/13087 [00:05<00:04, 1256.04it/s]","\rparsing log, completed traces ::  60%|#####9    | 7798/13087 [00:05<00:03, 1336.26it/s]","\rparsing log, completed traces ::  61%|######    | 7939/13087 [00:06<00:03, 1331.07it/s]","\rparsing log, completed traces ::  62%|######1   | 8097/13087 [00:06<00:03, 1395.68it/s]","\rparsing log, completed traces ::  63%|######3   | 8255/13087 [00:06<00:03, 1447.91it/s]","\rparsing log, completed traces ::  64%|######4   | 8404/13087 [00:06<00:03, 1458.83it/s]","\rparsing log, completed traces ::  65%|######5   | 8563/13087 [00:06<00:03, 1495.81it/s]","\rparsing log, completed traces ::  67%|######6   | 8720/13087 [00:06<00:02, 1514.54it/s]","\rparsing log, completed traces ::  68%|######7   | 8873/13087 [00:06<00:02, 1487.65it/s]","\rparsing log, completed traces ::  69%|######8   | 9023/13087 [00:06<00:02, 1484.16it/s]","\rparsing log, completed traces ::  70%|#######   | 9173/13087 [00:06<00:02, 1470.28it/s]","\rparsing log, completed traces ::  71%|#######1  | 9347/13087 [00:07<00:02, 1547.53it/s]","\rparsing log, completed traces ::  73%|#######2  | 9503/13087 [00:07<00:02, 1494.79it/s]","\rparsing log, completed traces ::  74%|#######3  | 9654/13087 [00:07<00:04, 745.68it/s] ","\rparsing log, completed traces ::  75%|#######5  | 9818/13087 [00:07<00:03, 897.72it/s]","\rparsing log, completed traces ::  76%|#######6  | 9957/13087 [00:07<00:03, 993.88it/s]","\rparsing log, completed traces ::  77%|#######7  | 10110/13087 [00:07<00:02, 1110.37it/s]","\rparsing log, completed traces ::  78%|#######8  | 10249/13087 [00:07<00:02, 1168.06it/s]","\rparsing log, completed traces ::  79%|#######9  | 10399/13087 [00:08<00:02, 1251.30it/s]","\rparsing log, completed traces ::  81%|########  | 10541/13087 [00:08<00:02, 1238.47it/s]","\rparsing log, completed traces ::  82%|########1 | 10692/13087 [00:08<00:01, 1308.06it/s]","\rparsing log, completed traces ::  83%|########2 | 10851/13087 [00:08<00:01, 1384.18it/s]","\rparsing log, completed traces ::  84%|########4 | 11026/13087 [00:08<00:01, 1483.88it/s]","\rparsing log, completed traces ::  85%|########5 | 11183/13087 [00:08<00:01, 1505.46it/s]","\rparsing log, completed traces ::  87%|########6 | 11349/13087 [00:08<00:01, 1550.05it/s]","\rparsing log, completed traces ::  88%|########8 | 11527/13087 [00:08<00:00, 1614.91it/s]","\rparsing log, completed traces ::  89%|########9 | 11694/13087 [00:08<00:00, 1630.48it/s]","\rparsing log, completed traces ::  91%|######### | 11875/13087 [00:09<00:00, 1677.26it/s]","\rparsing log, completed traces ::  92%|#########2| 12044/13087 [00:09<00:00, 1664.94it/s]","\rparsing log, completed traces ::  93%|#########3| 12217/13087 [00:09<00:00, 1682.33it/s]","\rparsing log, completed traces ::  95%|#########4| 12393/13087 [00:09<00:00, 1701.60it/s]","\rparsing log, completed traces ::  96%|#########6| 12577/13087 [00:09<00:00, 1739.91it/s]","\rparsing log, completed traces ::  97%|#########7| 12752/13087 [00:09<00:00, 1726.88it/s]","\rparsing log, completed traces ::  99%|#########8| 12925/13087 [00:09<00:00, 1724.36it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:09<00:00, 1349.86it/s]","\n","[data] Dataset BPI2012 shape=(164506, 5)","\n","[data] Built 25251 prefix samples from 4000 cases.","\n","[split] Train=18106 Val=2645 Test=4500","\n","[vocab] Activities: 19","\n","Epoch 1: validation_loss = 0.6370","\n","Epoch 2: validation_loss = 0.5793","\n","Epoch 3: validation_loss = 0.5342","\n","Epoch 4: validation_loss = 0.5251","\n","Epoch 5: validation_loss = 0.5232","\n","Epoch 6: validation_loss = 0.5166","\n","Epoch 7: validation_loss = 0.5133","\n","Epoch 8: validation_loss = 0.5241","\n","Epoch 9: validation_loss = 0.5094","\n","Epoch 10: validation_loss = 0.5175","\n","[final] Train: acc=0.7847 macroF1=0.5650 top3=0.9865 loss=0.4997","\n","[final] Val:   acc=0.7800 macroF1=0.5799 top3=0.9845 loss=0.5094","\n","[final] Test:  acc=0.7680 macroF1=0.5702 top3=0.9869 loss=0.5173","\n","Traceback (most recent call last):\n  File \"runfile.py\", line 558, in <module>\n    plot_curves(\n  File \"runfile.py\", line 356, in plot_curves\n    plt.plot(epochs, train_vals, label=\"train\")\n  File \"/workspace/.venv/lib/python3.10/site-packages/matplotlib/pyplot.py\", line 3838, in plot\n    return gca().plot(\n  File \"/workspace/.venv/lib/python3.10/site-packages/matplotlib/axes/_axes.py\", line 1777, in plot\n    lines = [*self._get_lines(self, *args, data=data, **kwargs)]\n  File \"/workspace/.venv/lib/python3.10/site-packages/matplotlib/axes/_base.py\", line 297, in __call__\n    yield from self._plot_args(\n  File \"/workspace/.venv/lib/python3.10/site-packages/matplotlib/axes/_base.py\", line 494, in _plot_args\n    raise ValueError(f\"x and y must have same first dimension, but \"\nValueError: x and y must have same first dimension, but have shapes (10,) and (11,)\n","Execution time: 45 seconds seconds (time limit is an hour)."],"parse_metrics_plan":"I will load the experiment_data.npy from the working directory and parse its nested structure. For each dataset key, I will extract the final metrics for train, validation, and test splits from the 'final' entries, falling back to the last epoch if needed. For losses, I will compute and print the best (minimum) training and validation loss. I will print the dataset name first, followed by clearly labeled metrics such as train accuracy, validation macro F1 score, and test top-3 accuracy. No plots will be generated.","parse_metrics_code":"import os\nimport numpy as np\n\n# Resolve working directory and load experiment data\nworking_dir = os.path.join(os.getcwd(), \"working\")\nexperiment_path = os.path.join(working_dir, \"experiment_data.npy\")\nexperiment_data = np.load(experiment_path, allow_pickle=True).item()\n\n\ndef extract_final_metrics(metrics_list):\n    \"\"\"\n    metrics_list is a list of tuples: (epoch_or_label, metrics_dict)\n    We prefer the entry with label 'final'. If not found, use the last element.\n    Returns a dict or an empty dict if none available.\n    \"\"\"\n    if not metrics_list:\n        return {}\n    # Try to find 'final'\n    for k, v in metrics_list:\n        if k == \"final\":\n            return v if isinstance(v, dict) else {}\n    # Fallback to last entry\n    last = metrics_list[-1]\n    if isinstance(last, tuple) and isinstance(last[1], dict):\n        return last[1]\n    return {}\n\n\ndef extract_best_loss(loss_list):\n    \"\"\"\n    loss_list is a list of tuples: (epoch, loss_value)\n    Returns best (minimum) loss if available, else None.\n    \"\"\"\n    if not loss_list:\n        return None\n    # Filter valid numeric losses\n    vals = [\n        lv for (_, lv) in loss_list if isinstance(lv, (int, float)) and np.isfinite(lv)\n    ]\n    if not vals:\n        return None\n    return float(min(vals))\n\n\ndef extract_final_loss(loss_list):\n    \"\"\"\n    Returns the last loss value if available, else None.\n    \"\"\"\n    if not loss_list:\n        return None\n    last = loss_list[-1][1]\n    return float(last) if isinstance(last, (int, float)) and np.isfinite(last) else None\n\n\ndef format_float(x, decimals=4):\n    try:\n        return f\"{float(x):.{decimals}f}\"\n    except Exception:\n        return \"nan\"\n\n\n# Iterate over datasets in experiment_data\nfor dataset_name, content in experiment_data.items():\n    print(dataset_name)\n\n    # Losses: print best training loss and best validation loss if available\n    losses = content.get(\"losses\", {})\n    train_losses = losses.get(\"train\", [])\n    val_losses = losses.get(\"val\", [])\n    best_train_loss = extract_best_loss(train_losses)\n    best_val_loss = extract_best_loss(val_losses)\n    if best_train_loss is not None:\n        print(f\"best training loss: {format_float(best_train_loss)}\")\n    if best_val_loss is not None:\n        print(f\"best validation loss: {format_float(best_val_loss)}\")\n\n    # Metrics: final metrics for train/val/test\n    metrics = content.get(\"metrics\", {})\n    train_metrics = extract_final_metrics(metrics.get(\"train\", []))\n    val_metrics = extract_final_metrics(metrics.get(\"val\", []))\n    test_metrics = extract_final_metrics(metrics.get(\"test\", []))\n\n    # Helper to print metrics with clear labels\n    def print_split_metrics(split_name, m):\n        if not m:\n            return\n        # Known keys from original code: accuracy, macro_f1, top3\n        if \"accuracy\" in m:\n            print(f\"{split_name} accuracy: {format_float(m['accuracy'])}\")\n        if \"macro_f1\" in m:\n            print(f\"{split_name} macro F1 score: {format_float(m['macro_f1'])}\")\n        if \"top3\" in m:\n            print(f\"{split_name} top-3 accuracy: {format_float(m['top3'])}\")\n\n    print_split_metrics(\"train\", train_metrics)\n    print_split_metrics(\"validation\", val_metrics)\n    print_split_metrics(\"test\", test_metrics)\n","parse_term_out":["PPM_NextActivity","\n","best training loss: 0.5149","\n","best validation loss: 0.5094","\n","train accuracy: 0.7847","\n","train macro F1 score: 0.5650","\n","train top-3 accuracy: 0.9865","\n","validation accuracy: 0.7800","\n","validation macro F1 score: 0.5799","\n","validation top-3 accuracy: 0.9845","\n","test accuracy: 0.7680","\n","test macro F1 score: 0.5702","\n","test top-3 accuracy: 0.9869","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":45.79446244239807,"exc_type":"ValueError","exc_info":{"args":["x and y must have same first dimension, but have shapes (10,) and (11,)"]},"exc_stack":[["/workspace/ai_scientist/treesearch/interpreter.py",144,"_run_session","exec(compile(code, self.agent_file_name, \"exec\"), global_scope)"],["runfile.py",558,"<module>","plot_curves("],["runfile.py",356,"plot_curves","plt.plot(epochs, train_vals, label=\"train\")"],["/workspace/.venv/lib/python3.10/site-packages/matplotlib/pyplot.py",3838,"plot","return gca().plot("],["/workspace/.venv/lib/python3.10/site-packages/matplotlib/axes/_axes.py",1777,"plot","lines = [*self._get_lines(self, *args, data=data, **kwargs)]"],["/workspace/.venv/lib/python3.10/site-packages/matplotlib/axes/_base.py",297,"__call__","yield from self._plot_args("],["/workspace/.venv/lib/python3.10/site-packages/matplotlib/axes/_base.py",494,"_plot_args","raise ValueError(f\"x and y must have same first dimension, but \""]],"analysis":"The training and evaluation ran successfully on the BPI 2012 log with a time-based split and produced reasonable metrics. However, the script crashed during plotting due to a mismatch between the x-axis (epochs) and y-axis (values) lengths. Specifically, for the Top-3 plot, train_top3 collects all entries from experiment_data['PPM_NextActivity']['metrics']['train'], including the extra ('final', ...) entry appended after training, yielding 11 points, while epochs_axis has 10 epochs. This results in ValueError: x and y must have same first dimension. Fix: filter out non-epoch entries when constructing plotting vectors, or move plotting before appending the 'final' entries. Example fixes:\n\n- For top-3 plotting:\n  train_top3 = [m[1]['top3'] for m in experiment_data['PPM_NextActivity']['metrics']['train'] if isinstance(m[0], int)]\n  val_top3 = [m[1]['top3'] for m in experiment_data['PPM_NextActivity']['metrics']['val'] if isinstance(m[0], int)]\n\n- Similarly, ensure losses lists or any plotted arrays are aligned with epochs, or compute epochs_axis as range(1, len(filtered_values)+1).\n\nAlternatively, perform plotting before appending the ('final', ...) metrics to avoid the mismatch.","exp_results_dir":null,"metric":{"value":{"metric_names":[{"metric_name":"best training loss","lower_is_better":true,"description":"Lowest observed loss on the training split during training.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.5149,"best_value":0.5149}]},{"metric_name":"best validation loss","lower_is_better":true,"description":"Lowest observed loss on the validation split during training.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.5094,"best_value":0.5094}]},{"metric_name":"train accuracy","lower_is_better":false,"description":"Final accuracy on the training split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.7847,"best_value":0.7847}]},{"metric_name":"train macro F1 score","lower_is_better":false,"description":"Final macro-averaged F1 score on the training split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.565,"best_value":0.565}]},{"metric_name":"train top-3 accuracy","lower_is_better":false,"description":"Final top-3 accuracy on the training split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.9865,"best_value":0.9865}]},{"metric_name":"validation accuracy","lower_is_better":false,"description":"Final accuracy on the validation split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.78,"best_value":0.78}]},{"metric_name":"validation macro F1 score","lower_is_better":false,"description":"Final macro-averaged F1 score on the validation split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.5799,"best_value":0.5799}]},{"metric_name":"validation top-3 accuracy","lower_is_better":false,"description":"Final top-3 accuracy on the validation split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.9845,"best_value":0.9845}]},{"metric_name":"test accuracy","lower_is_better":false,"description":"Final accuracy on the test split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.768,"best_value":0.768}]},{"metric_name":"test macro F1 score","lower_is_better":false,"description":"Final macro-averaged F1 score on the test split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.5702,"best_value":0.5702}]},{"metric_name":"test top-3 accuracy","lower_is_better":false,"description":"Final top-3 accuracy on the test split.","data":[{"dataset_name":"PPM_NextActivity","final_value":0.9869,"best_value":0.9869}]}]},"maximize":null,"name":null,"description":null},"is_buggy":true,"is_buggy_plots":null,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":[],"plot_paths":[],"plot_analyses":[],"vlm_feedback_summary":[],"datasets_successfully_tested":[],"ablation_name":null,"hyperparam_name":null,"is_seed_node":false,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"import os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nfrom pathlib import Path\nfrom collections import defaultdict\nfrom datetime import timezone\nimport torch\nimport torch.nn as nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import accuracy_score, f1_score\nimport warnings\n\nwarnings.filterwarnings(\"ignore\")\n\n# Device handling\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data structure\nexperiment_data = {\n    \"default_dataset\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n        \"meta\": {},\n    }\n}\n\n\n# ------------- Data Loading -------------\ndef load_local_xes():\n    try:\n        # Prefer helper if available\n        from ai_scientist.ideas.my_research_topic import (\n            load_datasets,\n            pick_default_dataset,\n        )\n\n        datasets = load_datasets()\n        name, df = pick_default_dataset(datasets)\n        return name, df\n    except Exception as e:\n        print(f\"[fallback] Helper loader failed: {e}\")\n        # Fallback: search input dir\n        try:\n            from pm4py.objects.log.importer.xes import importer as xes_importer\n        except Exception as e2:\n            raise ImportError(\"pm4py is required. Install: pip install pm4py\") from e2\n        # find any xes\n        base = Path(\"input\")\n        if not base.exists():\n            raise FileNotFoundError(\"No input directory with XES files found.\")\n        xes_files = list(base.glob(\"*.xes\")) + list(base.glob(\"*.xes.gz\"))\n        if not xes_files:\n            raise FileNotFoundError(\"No .xes or .xes.gz files in input directory.\")\n        # pick first\n        xes_path = xes_files[0]\n        print(f\"[fallback] Loading XES: {xes_path}\")\n        log = xes_importer.apply(str(xes_path))\n        rows = []\n        for tr in log:\n            case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n                \"case:concept:name\"\n            )\n            for e in tr:\n                rows.append(\n                    {\n                        \"case_id\": case_id,\n                        \"activity\": e.get(\"concept:name\"),\n                        \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                        \"timestamp\": e.get(\"time:timestamp\"),\n                        \"resource\": e.get(\"org:resource\", \"System\"),\n                    }\n                )\n        df = pd.DataFrame(rows)\n        df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n        df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n        df = df[[\"case_id\", \"activity\", \"lifecycle\", \"timestamp\", \"resource\"]]\n        df = df.sort_values([\"timestamp\", \"case_id\"]).reset_index(drop=True)\n        return xes_path.stem, df\n\n\ndataset_name, df = load_local_xes()\nprint(f\"[data] Using dataset: {dataset_name}, shape={df.shape}\")\n\n# ------------- Preprocessing & Prefix Generation -------------\n# Keep only 'complete' events if lifecycle exists\nif \"lifecycle\" in df.columns:\n    if df[\"lifecycle\"].notna().any():\n        # Many logs use 'complete'; if missing keep all\n        if (df[\"lifecycle\"] == \"complete\").any():\n            df = df[df[\"lifecycle\"] == \"complete\"].copy()\n\n# Sort within case by timestamp\ndf = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n\n# Build traces and case start times\ntraces = []\ncase_start_times = {}\nfor cid, g in df.groupby(\"case_id\"):\n    g = g.sort_values(\"timestamp\")\n    acts = g[\"activity\"].tolist()\n    times = pd.to_datetime(g[\"timestamp\"]).tolist()\n    if len(acts) >= 2:\n        traces.append((cid, acts, times))\n        case_start_times[cid] = times[0]\n\n# Time-based split at case level\ncases_sorted = sorted(traces, key=lambda x: case_start_times[x[0]])\nn = len(cases_sorted)\nif n == 0:\n    raise RuntimeError(\"No usable traces with length >= 2 found.\")\ntrain_end = int(0.7 * n)\nval_end = int(0.85 * n)\ntrain_cases = cases_sorted[:train_end]\nval_cases = cases_sorted[train_end:val_end]\ntest_cases = cases_sorted[val_end:]\n\nprint(\n    f\"[split] Cases: train={len(train_cases)}, val={len(val_cases)}, test={len(test_cases)}\"\n)\n\n# Build activity vocab from training only to avoid leakage\nfrom itertools import chain\n\ntrain_acts = list(chain.from_iterable([t[1] for t in train_cases]))\nact_counts = pd.Series(train_acts).value_counts()\nact_list = act_counts.index.tolist()\n# Add UNK for unseen in val/test\nPAD = \"<PAD>\"\nUNK = \"<UNK>\"\nitos = [PAD, UNK] + act_list\nstoi = {a: i for i, a in enumerate(itos)}\npad_idx = stoi[PAD]\nunk_idx = stoi[UNK]\nnum_classes = len(itos)\n\n\ndef act_to_idx(a):\n    return stoi.get(a, unk_idx)\n\n\n# Feature normalization helpers computed on train\ndef build_prefix_samples(cases):\n    samples = []\n    for cid, acts, times in cases:\n        # per-case time features relative to case start\n        t0 = times[0]\n        prev_time = None\n        # iterate prefixes\n        for t in range(1, len(acts)):  # prefix length t, target at t\n            prefix_acts = acts[:t]\n            prefix_times = times[:t]\n            target = acts[t]\n            # build per-event time features aligned with prefix positions\n            feats = []\n            for i, ts in enumerate(prefix_times):\n                delta_case = (ts - t0).total_seconds()\n                delta_prev = (\n                    (ts - prefix_times[i - 1]).total_seconds() if i > 0 else 0.0\n                )\n                dt = (\n                    pd.Timestamp(ts).tz_convert(\"UTC\")\n                    if pd.Timestamp(ts).tzinfo\n                    else pd.Timestamp(ts).tz_localize(\"UTC\")\n                )\n                hour = dt.hour\n                weekday = dt.weekday()\n                feats.append([delta_case, delta_prev, hour, weekday])\n            samples.append(\n                (\n                    cid,\n                    [act_to_idx(a) for a in prefix_acts],\n                    np.array(feats, dtype=np.float32),\n                    act_to_idx(target),\n                )\n            )\n    return samples\n\n\ntrain_samples = build_prefix_samples(train_cases)\nval_samples = build_prefix_samples(val_cases)\ntest_samples = build_prefix_samples(test_cases)\n\nprint(\n    f\"[samples] train={len(train_samples)}, val={len(val_samples)}, test={len(test_samples)}\"\n)\n\n\n# Fit normalizers on train for numerical features: delta_case, delta_prev, hour, weekday\ndef compute_norm_stats(samples):\n    X = (\n        np.concatenate([s[2] for s in samples], axis=0)\n        if samples\n        else np.zeros((0, 4), dtype=np.float32)\n    )\n    if X.shape[0] == 0:\n        return np.zeros(4, dtype=np.float32), np.ones(4, dtype=np.float32)\n    mean = X.mean(axis=0)\n    std = X.std(axis=0) + 1e-6\n    return mean.astype(np.float32), std.astype(np.float32)\n\n\nnum_mean, num_std = compute_norm_stats(train_samples)\n\n\ndef normalize_feats(feats):\n    # feats: [T,4]\n    f = feats.copy()\n    # Normalize seconds-based first two; scale hour and weekday to 0-1\n    f[:, 0:2] = (f[:, 0:2] - num_mean[0:2]) / num_std[0:2]\n    f[:, 2] = (f[:, 2] - num_mean[2]) / (num_std[2])\n    f[:, 3] = (f[:, 3] - num_mean[3]) / (num_std[3])\n    return f\n\n\n# ------------- Dataset & DataLoader -------------\nclass PrefixDataset(Dataset):\n    def __init__(self, samples):\n        self.samples = samples\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        cid, act_seq, feats, target = self.samples[idx]\n        feats = normalize_feats(feats)\n        return {\n            \"act_seq\": torch.tensor(act_seq, dtype=torch.long),\n            \"feats\": torch.tensor(feats, dtype=torch.float32),\n            \"target\": torch.tensor(target, dtype=torch.long),\n            \"length\": torch.tensor(len(act_seq), dtype=torch.long),\n        }\n\n\ndef collate_fn(batch):\n    # pad sequences\n    lengths = torch.tensor([b[\"length\"].item() for b in batch], dtype=torch.long)\n    max_len = lengths.max().item()\n    B = len(batch)\n    act_pad = torch.full((B, max_len), pad_idx, dtype=torch.long)\n    feats_pad = torch.zeros((B, max_len, 4), dtype=torch.float32)\n    targets = torch.tensor([b[\"target\"].item() for b in batch], dtype=torch.long)\n    for i, b in enumerate(batch):\n        l = b[\"length\"].item()\n        act_pad[i, :l] = b[\"act_seq\"]\n        feats_pad[i, :l, :] = b[\"feats\"]\n    return {\n        \"act_seq\": act_pad,\n        \"feats\": feats_pad,\n        \"lengths\": lengths,\n        \"target\": targets,\n    }\n\n\n# For speed, optionally cap number of samples\nmax_train_samples = 60000\nmax_val_samples = 15000\nmax_test_samples = 20000\nif len(train_samples) > max_train_samples:\n    train_samples = train_samples[:max_train_samples]\nif len(val_samples) > max_val_samples:\n    val_samples = val_samples[:max_val_samples]\nif len(test_samples) > max_test_samples:\n    test_samples = test_samples[:max_test_samples]\n\ntrain_ds = PrefixDataset(train_samples)\nval_ds = PrefixDataset(val_samples)\ntest_ds = PrefixDataset(test_samples)\n\nbatch_size = 128\ntrain_loader = DataLoader(\n    train_ds, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n)\nval_loader = DataLoader(\n    val_ds, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n)\ntest_loader = DataLoader(\n    test_ds, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n)\n\n\n# ------------- Model -------------\nclass LSTMNextActivity(nn.Module):\n    def __init__(\n        self,\n        vocab_size,\n        pad_idx,\n        emb_dim=64,\n        hidden_dim=128,\n        num_feat=4,\n        num_classes=0,\n        dropout=0.1,\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + num_feat,\n            hidden_size=hidden_dim,\n            num_layers=1,\n            batch_first=True,\n        )\n        self.dropout = nn.Dropout(dropout)\n        self.fc = nn.Linear(hidden_dim, num_classes)\n\n    def forward(self, act_seq, feats, lengths):\n        # act_seq: [B,T], feats: [B,T,4], lengths: [B]\n        emb = self.emb(act_seq)  # [B,T,E]\n        x = torch.cat([emb, feats], dim=-1)  # [B,T,E+4]\n        # pack\n        lengths_cpu = lengths.cpu()\n        packed = nn.utils.rnn.pack_padded_sequence(\n            x, lengths_cpu, batch_first=True, enforce_sorted=False\n        )\n        packed_out, (h_n, c_n) = self.lstm(packed)\n        # Use last hidden state from LSTM\n        h_last = h_n[-1]  # [B,H]\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)  # [B,C]\n        return logits\n\n\nmodel = LSTMNextActivity(\n    vocab_size=num_classes,\n    pad_idx=pad_idx,\n    emb_dim=64,\n    hidden_dim=128,\n    num_feat=4,\n    num_classes=num_classes,\n    dropout=0.2,\n).to(device)\ncriterion = nn.CrossEntropyLoss()\noptimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n\n\n# ------------- Metrics -------------\ndef topk_accuracy(logits, targets, k=3):\n    with torch.no_grad():\n        topk = torch.topk(logits, k=k, dim=1).indices\n        correct = (topk == targets.unsqueeze(1)).any(dim=1).float()\n        return correct.mean().item()\n\n\ndef evaluate(loader):\n    model.eval()\n    total_loss = 0.0\n    all_preds = []\n    all_targets = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"act_seq\"], batch[\"feats\"], batch[\"lengths\"])\n            loss = criterion(logits, batch[\"target\"])\n            total_loss += loss.item() * batch[\"target\"].size(0)\n            preds = logits.argmax(dim=1).cpu().numpy()\n            all_preds.append(preds)\n            all_targets.append(batch[\"target\"].cpu().numpy())\n    y_pred = np.concatenate(all_preds, axis=0) if all_preds else np.array([])\n    y_true = np.concatenate(all_targets, axis=0) if all_targets else np.array([])\n    avg_loss = total_loss / max(1, len(y_true))\n    acc = accuracy_score(y_true, y_pred) if y_true.size > 0 else 0.0\n    macro_f1 = (\n        f1_score(y_true, y_pred, average=\"macro\", zero_division=0)\n        if y_true.size > 0\n        else 0.0\n    )\n    # compute top-3 with a second pass for exactness\n    top3_acc = 0.0\n    if y_true.size > 0:\n        # recompute logits in mini-batches to get top-3\n        t3_correct = 0\n        n_total = 0\n        with torch.no_grad():\n            for batch in loader:\n                batch = {\n                    k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                    for k, v in batch.items()\n                }\n                logits = model(batch[\"act_seq\"], batch[\"feats\"], batch[\"lengths\"])\n                topk = torch.topk(logits, k=min(3, logits.size(1)), dim=1).indices\n                correct = (topk == batch[\"target\"].unsqueeze(1)).any(dim=1).sum().item()\n                t3_correct += correct\n                n_total += batch[\"target\"].size(0)\n        top3_acc = t3_correct / max(1, n_total)\n    return avg_loss, acc, macro_f1, top3_acc, y_pred, y_true\n\n\n# ------------- Training Loop -------------\nepochs = 8\nbest_val_loss = float(\"inf\")\nfor epoch in range(1, epochs + 1):\n    model.train()\n    running_loss = 0.0\n    for batch in train_loader:\n        batch = {\n            k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n            for k, v in batch.items()\n        }\n        optimizer.zero_grad()\n        logits = model(batch[\"act_seq\"], batch[\"feats\"], batch[\"lengths\"])\n        loss = criterion(logits, batch[\"target\"])\n        loss.backward()\n        optimizer.step()\n        running_loss += loss.item() * batch[\"target\"].size(0)\n    train_loss = running_loss / max(1, len(train_ds))\n\n    # Evaluate train/val\n    tr_loss_eval, tr_acc, tr_f1, tr_top3, _, _ = evaluate(train_loader)\n    val_loss, val_acc, val_f1, val_top3, _, _ = evaluate(val_loader)\n\n    print(\n        f\"Epoch {epoch}: train_loss = {train_loss:.4f} | train_acc = {tr_acc:.4f} | train_f1 = {tr_f1:.4f} | train_top3 = {tr_top3:.4f}\"\n    )\n    print(\n        f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc = {val_acc:.4f} | val_f1 = {val_f1:.4f} | val_top3 = {val_top3:.4f}\"\n    )\n\n    experiment_data[\"default_dataset\"][\"losses\"][\"train\"].append(\n        (epoch, float(train_loss))\n    )\n    experiment_data[\"default_dataset\"][\"losses\"][\"val\"].append((epoch, float(val_loss)))\n    experiment_data[\"default_dataset\"][\"metrics\"][\"train\"].append(\n        (epoch, float(tr_acc), float(tr_f1), float(tr_top3))\n    )\n    experiment_data[\"default_dataset\"][\"metrics\"][\"val\"].append(\n        (epoch, float(val_acc), float(val_f1), float(val_top3))\n    )\n    experiment_data[\"default_dataset\"][\"epochs\"].append(epoch)\n\n    if val_loss < best_val_loss:\n        best_val_loss = val_loss\n        # Save best model weights\n        torch.save(\n            model.state_dict(),\n            os.path.join(working_dir, f\"best_model_{dataset_name}.pt\"),\n        )\n\n# Load best and evaluate on test\nbest_path = os.path.join(working_dir, f\"best_model_{dataset_name}.pt\")\nif os.path.exists(best_path):\n    model.load_state_dict(torch.load(best_path, map_location=device))\n\ntest_loss, test_acc, test_f1, test_top3, y_pred_test, y_true_test = evaluate(\n    test_loader\n)\nprint(\n    f\"[final] Test: loss={test_loss:.4f} | acc={test_acc:.4f} | macro_f1={test_f1:.4f} | top3_acc={test_top3:.4f}\"\n)\n\nexperiment_data[\"default_dataset\"][\"metrics\"][\"test\"].append(\n    (\"final\", float(test_acc), float(test_f1), float(test_top3))\n)\nexperiment_data[\"default_dataset\"][\"predictions\"] = y_pred_test\nexperiment_data[\"default_dataset\"][\"ground_truth\"] = y_true_test\nexperiment_data[\"default_dataset\"][\"meta\"] = {\n    \"dataset_name\": dataset_name,\n    \"num_cases\": len(cases_sorted),\n    \"num_train_cases\": len(train_cases),\n    \"num_val_cases\": len(val_cases),\n    \"num_test_cases\": len(test_cases),\n    \"num_classes\": int(num_classes),\n    \"vocab_size\": int(num_classes),\n    \"primary_metric\": \"Top-3 Next-Activity Accuracy\",\n    \"test_top3\": float(test_top3),\n    \"test_accuracy\": float(test_acc),\n    \"test_macro_f1\": float(test_f1),\n}\n\n# Save experiment data\nnp.save(\n    os.path.join(working_dir, \"experiment_data.npy\"), experiment_data, allow_pickle=True\n)\n\n# Also print primary metric explicitly\nprint(f\"Top-3 Next-Activity Accuracy (test) = {test_top3:.4f}\")\n","plan":"We implement a minimal predictive process monitoring baseline for next-activity prediction using a 1-layer LSTM with simple process-aware features on local XES logs. We first discover and load a default dataset (preferring BPI 2017, then BPI 2012, then Road) via pm4py, then build case traces sorted by time. We generate prefix samples from each trace where the target is the immediate next activity, and features include embedded activity IDs and normalized time features (time since case start, time since last event, hour-of-day, weekday). We perform a strict time-based split at the case level into train/val/test (70/15/15) to avoid leakage. The model takes padded prefixes via PackedSequence, returns logits, and is trained with cross-entropy. We track per-epoch validation loss and metrics (accuracy, macro-F1, and the requested Top-3 accuracy) and report final results on train/val/test. Metrics, losses, predictions, and ground-truth are stored in experiment_data.npy under ./working, and we print Top-3 accuracy as the primary metric. For runtime safety we optionally cap the number of cases and set modest hyperparameters; all tensors and models are moved to GPU if available.","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport pandas as pd\nfrom pathlib import Path\nfrom collections import defaultdict\nfrom datetime import timezone\nimport torch\nimport torch.nn as nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import accuracy_score, f1_score, confusion_matrix\nimport warnings\n\nwarnings.filterwarnings(\"ignore\")\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\nexperiment_data = {\n    \"default_dataset\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n        \"meta\": {},\n    }\n}\n\n\ndef load_local_xes():\n    try:\n        from ai_scientist.ideas.my_research_topic import (\n            load_datasets,\n            pick_default_dataset,\n        )\n\n        datasets = load_datasets()\n        name, df = pick_default_dataset(datasets)\n        return name, df\n    except Exception as e:\n        print(f\"[fallback] Helper loader failed: {e}\")\n        try:\n            from pm4py.objects.log.importer.xes import importer as xes_importer\n        except Exception as e2:\n            raise ImportError(\"pm4py is required. Install: pip install pm4py\") from e2\n        base = Path(\"input\")\n        if not base.exists():\n            raise FileNotFoundError(\"No input directory with XES files found.\")\n        xes_files = list(base.glob(\"*.xes\")) + list(base.glob(\"*.xes.gz\"))\n        if not xes_files:\n            raise FileNotFoundError(\"No .xes or .xes.gz files in input directory.\")\n        xes_path = xes_files[0]\n        print(f\"[fallback] Loading XES: {xes_path}\")\n        log = xes_importer.apply(str(xes_path))\n        rows = []\n        for tr in log:\n            case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n                \"case:concept:name\"\n            )\n            for e in tr:\n                rows.append(\n                    {\n                        \"case_id\": case_id,\n                        \"activity\": e.get(\"concept:name\"),\n                        \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                        \"timestamp\": e.get(\"time:timestamp\"),\n                        \"resource\": e.get(\"org:resource\", \"System\"),\n                    }\n                )\n        df = pd.DataFrame(rows)\n        df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n        df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n        df = df[[\"case_id\", \"activity\", \"lifecycle\", \"timestamp\", \"resource\"]]\n        df = df.sort_values([\"timestamp\", \"case_id\"]).reset_index(drop=True)\n        return xes_path.stem, df\n\n\ndataset_name, df = load_local_xes()\nprint(f\"[data] Using dataset: {dataset_name}, shape={df.shape}\")\n\nif \"lifecycle\" in df.columns:\n    if df[\"lifecycle\"].notna().any():\n        if (df[\"lifecycle\"] == \"complete\").any():\n            df = df[df[\"lifecycle\"] == \"complete\"].copy()\n\ndf = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n\ntraces = []\ncase_start_times = {}\nfor cid, g in df.groupby(\"case_id\"):\n    g = g.sort_values(\"timestamp\")\n    acts = g[\"activity\"].tolist()\n    times = pd.to_datetime(g[\"timestamp\"]).tolist()\n    if len(acts) >= 2:\n        traces.append((cid, acts, times))\n        case_start_times[cid] = times[0]\n\ncases_sorted = sorted(traces, key=lambda x: case_start_times[x[0]])\nn = len(cases_sorted)\nif n == 0:\n    raise RuntimeError(\"No usable traces with length >= 2 found.\")\ntrain_end = int(0.7 * n)\nval_end = int(0.85 * n)\ntrain_cases = cases_sorted[:train_end]\nval_cases = cases_sorted[train_end:val_end]\ntest_cases = cases_sorted[val_end:]\nprint(\n    f\"[split] Cases: train={len(train_cases)}, val={len(val_cases)}, test={len(test_cases)}\"\n)\n\nfrom itertools import chain\n\ntrain_acts = list(chain.from_iterable([t[1] for t in train_cases]))\nact_counts = pd.Series(train_acts).value_counts()\nact_list = act_counts.index.tolist()\nPAD = \"<PAD>\"\nUNK = \"<UNK>\"\nitos = [PAD, UNK] + act_list\nstoi = {a: i for i, a in enumerate(itos)}\npad_idx = stoi[PAD]\nunk_idx = stoi[UNK]\nnum_classes = len(itos)\n\n\ndef act_to_idx(a):\n    return stoi.get(a, unk_idx)\n\n\ndef build_prefix_samples(cases):\n    samples = []\n    for cid, acts, times in cases:\n        t0 = times[0]\n        for t in range(1, len(acts)):\n            prefix_acts = acts[:t]\n            prefix_times = times[:t]\n            target = acts[t]\n            feats = []\n            for i, ts in enumerate(prefix_times):\n                delta_case = (ts - t0).total_seconds()\n                delta_prev = (\n                    (ts - prefix_times[i - 1]).total_seconds() if i > 0 else 0.0\n                )\n                dt = pd.Timestamp(ts)\n                if dt.tzinfo is None:\n                    dt = dt.tz_localize(\"UTC\")\n                else:\n                    dt = dt.tz_convert(\"UTC\")\n                feats.append([delta_case, delta_prev, dt.hour, dt.weekday()])\n            samples.append(\n                (\n                    cid,\n                    [act_to_idx(a) for a in prefix_acts],\n                    np.array(feats, dtype=np.float32),\n                    act_to_idx(target),\n                )\n            )\n    return samples\n\n\ntrain_samples = build_prefix_samples(train_cases)\nval_samples = build_prefix_samples(val_cases)\ntest_samples = build_prefix_samples(test_cases)\nprint(\n    f\"[samples] train={len(train_samples)}, val={len(val_samples)}, test={len(test_samples)}\"\n)\n\n\ndef compute_norm_stats(samples):\n    X = (\n        np.concatenate([s[2] for s in samples], axis=0)\n        if samples\n        else np.zeros((0, 4), dtype=np.float32)\n    )\n    if X.shape[0] == 0:\n        return np.zeros(4, dtype=np.float32), np.ones(4, dtype=np.float32)\n    mean = X.mean(axis=0)\n    std = X.std(axis=0) + 1e-6\n    return mean.astype(np.float32), std.astype(np.float32)\n\n\nnum_mean, num_std = compute_norm_stats(train_samples)\n\n\ndef normalize_feats(f):\n    f = f.copy()\n    f[:, 0:2] = (f[:, 0:2] - num_mean[0:2]) / num_std[0:2]\n    f[:, 2] = (f[:, 2] - num_mean[2]) / num_std[2]\n    f[:, 3] = (f[:, 3] - num_mean[3]) / num_std[3]\n    return f\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples):\n        self.samples = samples\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        _, act_seq, feats, target = self.samples[idx]\n        feats = normalize_feats(feats)\n        return {\n            \"act_seq\": torch.tensor(act_seq, dtype=torch.long),\n            \"feats\": torch.tensor(feats, dtype=torch.float32),\n            \"target\": torch.tensor(target, dtype=torch.long),\n            \"length\": torch.tensor(len(act_seq), dtype=torch.long),\n        }\n\n\ndef collate_fn(batch):\n    lengths = torch.tensor([b[\"length\"].item() for b in batch], dtype=torch.long)\n    max_len = lengths.max().item()\n    B = len(batch)\n    act_pad = torch.full((B, max_len), pad_idx, dtype=torch.long)\n    feats_pad = torch.zeros((B, max_len, 4), dtype=torch.float32)\n    targets = torch.tensor([b[\"target\"].item() for b in batch], dtype=torch.long)\n    for i, b in enumerate(batch):\n        l = b[\"length\"].item()\n        act_pad[i, :l] = b[\"act_seq\"]\n        feats_pad[i, :l, :] = b[\"feats\"]\n    return {\n        \"act_seq\": act_pad,\n        \"feats\": feats_pad,\n        \"lengths\": lengths,\n        \"target\": targets,\n    }\n\n\nmax_train_samples = 60000\nmax_val_samples = 15000\nmax_test_samples = 20000\nif len(train_samples) > max_train_samples:\n    train_samples = train_samples[:max_train_samples]\nif len(val_samples) > max_val_samples:\n    val_samples = val_samples[:max_val_samples]\nif len(test_samples) > max_test_samples:\n    test_samples = test_samples[:max_test_samples]\n\ntrain_ds = PrefixDataset(train_samples)\nval_ds = PrefixDataset(val_samples)\ntest_ds = PrefixDataset(test_samples)\nbatch_size = 128\ntrain_loader = DataLoader(\n    train_ds, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n)\nval_loader = DataLoader(\n    val_ds, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n)\ntest_loader = DataLoader(\n    test_ds, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n)\n\n\nclass LSTMNextActivity(nn.Module):\n    def __init__(\n        self,\n        vocab_size,\n        pad_idx,\n        emb_dim=64,\n        hidden_dim=128,\n        num_feat=4,\n        num_classes=0,\n        dropout=0.2,\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + num_feat,\n            hidden_size=hidden_dim,\n            num_layers=1,\n            batch_first=True,\n        )\n        self.dropout = nn.Dropout(dropout)\n        self.fc = nn.Linear(hidden_dim, num_classes)\n\n    def forward(self, act_seq, feats, lengths):\n        emb = self.emb(act_seq)\n        x = torch.cat([emb, feats], dim=-1)\n        packed = nn.utils.rnn.pack_padded_sequence(\n            x, lengths.cpu(), batch_first=True, enforce_sorted=False\n        )\n        _, (h_n, _) = self.lstm(packed)\n        h_last = self.dropout(h_n[-1])\n        return self.fc(h_last)\n\n\nmodel = LSTMNextActivity(num_classes, pad_idx).to(device)\ncriterion = nn.CrossEntropyLoss()\noptimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n\n\ndef evaluate(loader):\n    model.eval()\n    total_loss = 0.0\n    all_preds = []\n    all_targets = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"act_seq\"], batch[\"feats\"], batch[\"lengths\"])\n            loss = criterion(logits, batch[\"target\"])\n            total_loss += loss.item() * batch[\"target\"].size(0)\n            all_preds.append(logits.argmax(1).cpu().numpy())\n            all_targets.append(batch[\"target\"].cpu().numpy())\n    y_pred = np.concatenate(all_preds) if all_preds else np.array([])\n    y_true = np.concatenate(all_targets) if all_targets else np.array([])\n    avg_loss = total_loss / max(1, len(y_true))\n    acc = accuracy_score(y_true, y_pred) if y_true.size > 0 else 0.0\n    macro_f1 = (\n        f1_score(y_true, y_pred, average=\"macro\", zero_division=0)\n        if y_true.size > 0\n        else 0.0\n    )\n    # top-3\n    top3 = 0.0\n    if y_true.size > 0:\n        t3 = 0\n        n = 0\n        with torch.no_grad():\n            for batch in loader:\n                batch = {\n                    k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                    for k, v in batch.items()\n                }\n                logits = model(batch[\"act_seq\"], batch[\"feats\"], batch[\"lengths\"])\n                topk = torch.topk(logits, k=min(3, logits.size(1)), dim=1).indices\n                t3 += (topk == batch[\"target\"].unsqueeze(1)).any(1).sum().item()\n                n += batch[\"target\"].size(0)\n        top3 = t3 / max(1, n)\n    return avg_loss, acc, macro_f1, top3, y_pred, y_true\n\n\nepochs = 8\nbest_val_loss = float(\"inf\")\nfor epoch in range(1, epochs + 1):\n    model.train()\n    running = 0.0\n    for batch in train_loader:\n        batch = {\n            k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n            for k, v in batch.items()\n        }\n        optimizer.zero_grad()\n        logits = model(batch[\"act_seq\"], batch[\"feats\"], batch[\"lengths\"])\n        loss = criterion(logits, batch[\"target\"])\n        loss.backward()\n        optimizer.step()\n        running += loss.item() * batch[\"target\"].size(0)\n    train_loss = running / max(1, len(train_ds))\n    tr_loss_eval, tr_acc, tr_f1, tr_top3, _, _ = evaluate(train_loader)\n    val_loss, val_acc, val_f1, val_top3, _, _ = evaluate(val_loader)\n    print(\n        f\"Epoch {epoch}: train_loss = {train_loss:.4f} | train_acc = {tr_acc:.4f} | train_f1 = {tr_f1:.4f} | train_top3 = {tr_top3:.4f}\"\n    )\n    print(\n        f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc = {val_acc:.4f} | val_f1 = {val_f1:.4f} | val_top3 = {val_top3:.4f}\"\n    )\n    experiment_data[\"default_dataset\"][\"losses\"][\"train\"].append(\n        (epoch, float(train_loss))\n    )\n    experiment_data[\"default_dataset\"][\"losses\"][\"val\"].append((epoch, float(val_loss)))\n    experiment_data[\"default_dataset\"][\"metrics\"][\"train\"].append(\n        (epoch, float(tr_acc), float(tr_f1), float(tr_top3))\n    )\n    experiment_data[\"default_dataset\"][\"metrics\"][\"val\"].append(\n        (epoch, float(val_acc), float(val_f1), float(val_top3))\n    )\n    experiment_data[\"default_dataset\"][\"epochs\"].append(epoch)\n    if val_loss < best_val_loss:\n        best_val_loss = val_loss\n        torch.save(\n            model.state_dict(),\n            os.path.join(working_dir, f\"best_model_{dataset_name}.pt\"),\n        )\n\nbest_path = os.path.join(working_dir, f\"best_model_{dataset_name}.pt\")\nif os.path.exists(best_path):\n    model.load_state_dict(\n        torch.load(best_path, map_mode=device, map_location=device)\n        if hasattr(torch.load, \"__call__\")\n        else torch.load(best_path, map_location=device)\n    )\n\ntest_loss, test_acc, test_f1, test_top3, y_pred_test, y_true_test = evaluate(\n    test_loader\n)\nprint(\n    f\"[final] Test: loss={test_loss:.4f} | acc={test_acc:.4f} | macro_f1={test_f1:.4f} | top3_acc={test_top3:.4f}\"\n)\nexperiment_data[\"default_dataset\"][\"metrics\"][\"test\"].append(\n    (\"final\", float(test_acc), float(test_f1), float(test_top3))\n)\nexperiment_data[\"default_dataset\"][\"predictions\"] = y_pred_test\nexperiment_data[\"default_dataset\"][\"ground_truth\"] = y_true_test\nexperiment_data[\"default_dataset\"][\"meta\"] = {\n    \"dataset_name\": dataset_name,\n    \"num_cases\": len(cases_sorted),\n    \"num_train_cases\": len(train_cases),\n    \"num_val_cases\": len(val_cases),\n    \"num_test_cases\": len(test_cases),\n    \"num_classes\": int(num_classes),\n    \"vocab_size\": int(num_classes),\n    \"primary_metric\": \"Top-3 Next-Activity Accuracy\",\n    \"test_top3\": float(test_top3),\n    \"test_accuracy\": float(test_acc),\n    \"test_macro_f1\": float(test_f1),\n}\nnp.save(\n    os.path.join(working_dir, \"experiment_data.npy\"), experiment_data, allow_pickle=True\n)\nprint(f\"Top-3 Next-Activity Accuracy (test) = {test_top3:.4f}\")\n\n# ---------- Plotting from experiment_data.npy only ----------\ntry:\n    experiment_data = np.load(\n        os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n    ).item()\nexcept Exception as e:\n    print(f\"Error loading experiment data: {e}\")\n    experiment_data = None\n\nif experiment_data is not None:\n    ds_key = \"default_dataset\"\n    meta = experiment_data.get(ds_key, {}).get(\"meta\", {})\n    dname = meta.get(\"dataset_name\", \"dataset\")\n    # 1) Loss curves\n    try:\n        plt.figure()\n        losses = experiment_data[ds_key][\"losses\"]\n        if losses[\"train\"]:\n            ep, tr = zip(*losses[\"train\"])\n            plt.plot(ep, tr, label=\"Train Loss\")\n        if losses[\"val\"]:\n            epv, vl = zip(*losses[\"val\"])\n            plt.plot(epv, vl, label=\"Val Loss\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.title(f\"{dname} - Training/Validation Loss\\nNext-Activity Prediction (BPM)\")\n        plt.legend()\n        out = os.path.join(working_dir, f\"{dname}_loss_curves_next_activity.png\")\n        plt.savefig(out)\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating loss curves: {e}\")\n        plt.close()\n\n    # 2) Metric curves (Acc, Macro-F1, Top-3)\n    try:\n        plt.figure()\n        mtr = experiment_data[ds_key][\"metrics\"][\"train\"]\n        mval = experiment_data[ds_key][\"metrics\"][\"val\"]\n        if mtr:\n            ep_t, acc_t, f1_t, top3_t = zip(*mtr)\n            plt.plot(ep_t, acc_t, label=\"Train Acc\")\n            plt.plot(ep_t, f1_t, label=\"Train Macro-F1\")\n            plt.plot(ep_t, top3_t, label=\"Train Top-3\")\n        if mval:\n            ep_v, acc_v, f1_v, top3_v = zip(*mval)\n            plt.plot(ep_v, acc_v, \"--\", label=\"Val Acc\")\n            plt.plot(ep_v, f1_v, \"--\", label=\"Val Macro-F1\")\n            plt.plot(ep_v, top3_v, \"--\", label=\"Val Top-3\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Score\")\n        plt.title(\n            f\"{dname} - Training/Validation Metrics\\nNext-Activity Prediction (BPM)\"\n        )\n        plt.legend()\n        out = os.path.join(working_dir, f\"{dname}_metric_curves_next_activity.png\")\n        plt.savefig(out)\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating metric curves: {e}\")\n        plt.close()\n\n    # 3) Confusion Matrix (Test)\n    try:\n        y_true = experiment_data[ds_key].get(\"ground_truth\", [])\n        y_pred = experiment_data[ds_key].get(\"predictions\", [])\n        y_true = np.array(y_true)\n        y_pred = np.array(y_pred)\n        if y_true.size > 0 and y_pred.size > 0:\n            cm = confusion_matrix(y_true, y_pred, labels=np.unique(y_true))\n            plt.figure(figsize=(6, 5))\n            im = plt.imshow(cm, interpolation=\"nearest\", cmap=\"Blues\")\n            plt.colorbar(im)\n            plt.title(\n                f\"{dname} - Confusion Matrix (Test)\\nNext-Activity Prediction (BPM)\"\n            )\n            plt.xlabel(\"Predicted\")\n            plt.ylabel(\"True\")\n            ticks = np.arange(len(np.unique(y_true)))\n            plt.xticks(ticks, ticks, rotation=90)\n            plt.yticks(ticks, ticks)\n            plt.tight_layout()\n            out = os.path.join(\n                working_dir, f\"{dname}_confusion_matrix_next_activity.png\"\n            )\n            plt.savefig(out)\n            plt.close()\n        else:\n            print(\"Confusion Matrix skipped: no predictions/ground truth available.\")\n    except Exception as e:\n        print(f\"Error creating confusion matrix: {e}\")\n        plt.close()\n","plot_plan":"We will load a local XES log using pm4py, parse it into a dataframe, and create case-level time-based splits (70/15/15) to avoid temporal leakage. From training cases, we build an activity vocabulary (with PAD/UNK) and generate prefix samples for next-activity prediction, including simple time features per event. We normalize numerical features based on training data only and prepare PyTorch datasets/loaders with padded batches. A compact LSTM model consumes embedded activities concatenated with time features to predict the next activity. We train for a few epochs, tracking train/val losses and BPM metrics: accuracy, macro-F1, and Top-3 accuracy, selecting the best model by validation loss. We then evaluate on the test set and save all results into experiment_data.npy along with meta information. Finally, we load experiment_data.npy and, strictly using its contents, produce standard plots: training/validation loss curves, training/validation metrics curves, and a test-set confusion matrix for next-activity prediction. All plots are saved under working_dir with clear titles/subtitles and robust try-except blocks, and figures are always closed after saving.","step":2,"id":"3645f80923974ef6a45cb404d8ddc8bd","ctime":1757756009.5527756,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available in /workspace/data: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<25:40,  8.50it/s]","\rparsing log, completed traces ::   1%|1         | 136/13087 [00:00<00:17, 742.38it/s]","\rparsing log, completed traces ::   2%|2         | 274/13087 [00:00<00:12, 1018.87it/s]","\rparsing log, completed traces ::   3%|3         | 402/13087 [00:00<00:11, 1110.02it/s]","\rparsing log, completed traces ::   4%|4         | 540/13087 [00:00<00:10, 1203.19it/s]","\rparsing log, completed traces ::   5%|5         | 703/13087 [00:00<00:09, 1345.81it/s]","\rparsing log, completed traces ::   7%|6         | 854/13087 [00:00<00:08, 1396.69it/s]","\rparsing log, completed traces ::   8%|7         | 997/13087 [00:00<00:08, 1406.47it/s]","\rparsing log, completed traces ::   9%|8         | 1139/13087 [00:00<00:08, 1381.01it/s]","\rparsing log, completed traces ::  10%|9         | 1278/13087 [00:01<00:08, 1379.76it/s]","\rparsing log, completed traces ::  11%|#1        | 1458/13087 [00:01<00:07, 1500.60it/s]","\rparsing log, completed traces ::  12%|#2        | 1620/13087 [00:01<00:07, 1534.68it/s]","\rparsing log, completed traces ::  14%|#3        | 1776/13087 [00:01<00:07, 1541.18it/s]","\rparsing log, completed traces ::  15%|#4        | 1931/13087 [00:01<00:13, 839.43it/s] ","\rparsing log, completed traces ::  16%|#5        | 2066/13087 [00:01<00:11, 935.75it/s]","\rparsing log, completed traces ::  17%|#6        | 2215/13087 [00:01<00:10, 1051.66it/s]","\rparsing log, completed traces ::  18%|#8        | 2369/13087 [00:02<00:09, 1163.12it/s]","\rparsing log, completed traces ::  19%|#9        | 2524/13087 [00:02<00:08, 1249.83it/s]","\rparsing log, completed traces ::  20%|##        | 2672/13087 [00:02<00:07, 1307.43it/s]","\rparsing log, completed traces ::  22%|##1       | 2816/13087 [00:02<00:07, 1315.03it/s]","\rparsing log, completed traces ::  23%|##2       | 2957/13087 [00:02<00:07, 1316.01it/s]","\rparsing log, completed traces ::  24%|##3       | 3095/13087 [00:02<00:07, 1328.91it/s]","\rparsing log, completed traces ::  25%|##4       | 3253/13087 [00:02<00:07, 1398.37it/s]","\rparsing log, completed traces ::  26%|##6       | 3462/13087 [00:02<00:06, 1597.52it/s]","\rparsing log, completed traces ::  28%|##7       | 3628/13087 [00:02<00:05, 1602.61it/s]","\rparsing log, completed traces ::  29%|##8       | 3791/13087 [00:02<00:05, 1577.02it/s]","\rparsing log, completed traces ::  30%|###       | 3951/13087 [00:03<00:05, 1578.31it/s]","\rparsing log, completed traces ::  31%|###1      | 4110/13087 [00:03<00:10, 885.14it/s] ","\rparsing log, completed traces ::  33%|###2      | 4313/13087 [00:03<00:07, 1101.31it/s]","\rparsing log, completed traces ::  34%|###4      | 4492/13087 [00:03<00:06, 1246.55it/s]","\rparsing log, completed traces ::  36%|###5      | 4667/13087 [00:03<00:06, 1362.01it/s]","\rparsing log, completed traces ::  37%|###6      | 4829/13087 [00:03<00:05, 1398.08it/s]","\rparsing log, completed traces ::  38%|###8      | 5004/13087 [00:03<00:05, 1487.99it/s]","\rparsing log, completed traces ::  39%|###9      | 5168/13087 [00:04<00:05, 1474.99it/s]","\rparsing log, completed traces ::  41%|####      | 5326/13087 [00:04<00:05, 1502.86it/s]","\rparsing log, completed traces ::  42%|####1     | 5484/13087 [00:04<00:05, 1439.61it/s]","\rparsing log, completed traces ::  43%|####3     | 5634/13087 [00:04<00:05, 1435.52it/s]","\rparsing log, completed traces ::  44%|####4     | 5789/13087 [00:04<00:04, 1464.34it/s]","\rparsing log, completed traces ::  45%|####5     | 5945/13087 [00:04<00:04, 1480.98it/s]","\rparsing log, completed traces ::  47%|####6     | 6096/13087 [00:04<00:04, 1456.52it/s]","\rparsing log, completed traces ::  48%|####7     | 6259/13087 [00:04<00:04, 1501.54it/s]","\rparsing log, completed traces ::  49%|####9     | 6439/13087 [00:04<00:04, 1585.67it/s]","\rparsing log, completed traces ::  50%|#####     | 6599/13087 [00:04<00:04, 1491.21it/s]","\rparsing log, completed traces ::  52%|#####1    | 6750/13087 [00:05<00:04, 1476.63it/s]","\rparsing log, completed traces ::  53%|#####2    | 6899/13087 [00:05<00:08, 758.86it/s] ","\rparsing log, completed traces ::  54%|#####3    | 7040/13087 [00:05<00:06, 870.29it/s]","\rparsing log, completed traces ::  55%|#####4    | 7183/13087 [00:05<00:06, 978.68it/s]","\rparsing log, completed traces ::  56%|#####5    | 7326/13087 [00:05<00:05, 1077.51it/s]","\rparsing log, completed traces ::  57%|#####6    | 7459/13087 [00:05<00:05, 1112.77it/s]","\rparsing log, completed traces ::  58%|#####7    | 7590/13087 [00:06<00:04, 1160.85it/s]","\rparsing log, completed traces ::  59%|#####8    | 7720/13087 [00:06<00:04, 1195.46it/s]","\rparsing log, completed traces ::  60%|######    | 7871/13087 [00:06<00:04, 1277.98it/s]","\rparsing log, completed traces ::  61%|######1   | 8007/13087 [00:06<00:04, 1265.27it/s]","\rparsing log, completed traces ::  62%|######2   | 8160/13087 [00:06<00:03, 1339.34it/s]","\rparsing log, completed traces ::  64%|######3   | 8317/13087 [00:06<00:03, 1402.70it/s]","\rparsing log, completed traces ::  65%|######4   | 8465/13087 [00:06<00:03, 1416.87it/s]","\rparsing log, completed traces ::  66%|######5   | 8617/13087 [00:06<00:03, 1437.83it/s]","\rparsing log, completed traces ::  67%|######7   | 8770/13087 [00:06<00:02, 1462.38it/s]","\rparsing log, completed traces ::  68%|######8   | 8918/13087 [00:06<00:02, 1415.71it/s]","\rparsing log, completed traces ::  69%|######9   | 9061/13087 [00:07<00:02, 1368.38it/s]","\rparsing log, completed traces ::  70%|#######   | 9213/13087 [00:07<00:02, 1407.32it/s]","\rparsing log, completed traces ::  72%|#######1  | 9378/13087 [00:07<00:02, 1474.74it/s]","\rparsing log, completed traces ::  73%|#######2  | 9527/13087 [00:07<00:02, 1417.44it/s]","\rparsing log, completed traces ::  74%|#######3  | 9670/13087 [00:07<00:04, 715.44it/s] ","\rparsing log, completed traces ::  75%|#######5  | 9826/13087 [00:07<00:03, 857.54it/s]","\rparsing log, completed traces ::  76%|#######6  | 9960/13087 [00:08<00:03, 951.54it/s]","\rparsing log, completed traces ::  77%|#######7  | 10107/13087 [00:08<00:02, 1062.01it/s]","\rparsing log, completed traces ::  78%|#######8  | 10239/13087 [00:08<00:02, 1105.52it/s]","\rparsing log, completed traces ::  79%|#######9  | 10389/13087 [00:08<00:02, 1200.60it/s]","\rparsing log, completed traces ::  80%|########  | 10525/13087 [00:08<00:02, 1181.80it/s]","\rparsing log, completed traces ::  82%|########1 | 10669/13087 [00:08<00:01, 1248.28it/s]","\rparsing log, completed traces ::  83%|########2 | 10825/13087 [00:08<00:01, 1327.56it/s]","\rparsing log, completed traces ::  84%|########4 | 10996/13087 [00:08<00:01, 1431.66it/s]","\rparsing log, completed traces ::  85%|########5 | 11148/13087 [00:08<00:01, 1454.96it/s]","\rparsing log, completed traces ::  86%|########6 | 11304/13087 [00:08<00:01, 1472.42it/s]","\rparsing log, completed traces ::  88%|########7 | 11487/13087 [00:09<00:01, 1570.82it/s]","\rparsing log, completed traces ::  89%|########8 | 11647/13087 [00:09<00:00, 1529.00it/s]","\rparsing log, completed traces ::  90%|######### | 11825/13087 [00:09<00:00, 1599.18it/s]","\rparsing log, completed traces ::  92%|#########1| 11998/13087 [00:09<00:00, 1632.47it/s]","\rparsing log, completed traces ::  93%|#########2| 12163/13087 [00:09<00:00, 1622.97it/s]","\rparsing log, completed traces ::  94%|#########4| 12343/13087 [00:09<00:00, 1674.51it/s]","\rparsing log, completed traces ::  96%|#########5| 12516/13087 [00:09<00:00, 1690.50it/s]","\rparsing log, completed traces ::  97%|#########6| 12692/13087 [00:09<00:00, 1710.82it/s]","\rparsing log, completed traces ::  98%|#########8| 12864/13087 [00:09<00:00, 1696.67it/s]","\rparsing log, completed traces :: 100%|#########9| 13037/13087 [00:10<00:00, 1705.74it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:10<00:00, 1304.21it/s]","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<5:29:51,  1.59it/s]","\rparsing log, completed traces ::   0%|          | 59/31509 [00:00<04:49, 108.45it/s]","\rparsing log, completed traces ::   0%|          | 108/31509 [00:00<02:47, 187.98it/s]","\rparsing log, completed traces ::   1%|          | 165/31509 [00:00<01:54, 274.08it/s]","\rparsing log, completed traces ::   1%|          | 224/31509 [00:01<01:29, 351.35it/s]","\rparsing log, completed traces ::   1%|          | 277/31509 [00:01<01:18, 397.26it/s]","\rparsing log, completed traces ::   1%|1         | 335/31509 [00:01<01:10, 444.92it/s]","\rparsing log, completed traces ::   1%|1         | 390/31509 [00:01<01:05, 472.08it/s]","\rparsing log, completed traces ::   1%|1         | 444/31509 [00:01<01:04, 484.11it/s]","\rparsing log, completed traces ::   2%|1         | 502/31509 [00:01<01:00, 511.09it/s]","\rparsing log, completed traces ::   2%|1         | 557/31509 [00:01<00:59, 515.96it/s]","\rparsing log, completed traces ::   2%|1         | 614/31509 [00:01<00:58, 530.06it/s]","\rparsing log, completed traces ::   2%|2         | 680/31509 [00:01<00:54, 565.78it/s]","\rparsing log, completed traces ::   2%|2         | 738/31509 [00:01<00:54, 566.08it/s]","\rparsing log, completed traces ::   3%|2         | 796/31509 [00:02<01:35, 320.75it/s]","\rparsing log, completed traces ::   3%|2         | 851/31509 [00:02<01:24, 363.05it/s]","\rparsing log, completed traces ::   3%|2         | 908/31509 [00:02<01:15, 407.19it/s]","\rparsing log, completed traces ::   3%|3         | 972/31509 [00:02<01:06, 461.54it/s]","\rparsing log, completed traces ::   3%|3         | 1032/31509 [00:02<01:01, 494.06it/s]","\rparsing log, completed traces ::   3%|3         | 1088/31509 [00:02<01:00, 500.69it/s]","\rparsing log, completed traces ::   4%|3         | 1149/31509 [00:02<00:57, 526.88it/s]","\rparsing log, completed traces ::   4%|3         | 1212/31509 [00:03<00:54, 551.20it/s]","\rparsing log, completed traces ::   4%|4         | 1276/31509 [00:03<00:52, 572.78it/s]","\rparsing log, completed traces ::   4%|4         | 1336/31509 [00:03<00:52, 576.50it/s]","\rparsing log, completed traces ::   4%|4         | 1395/31509 [00:03<00:51, 579.48it/s]","\rparsing log, completed traces ::   5%|4         | 1454/31509 [00:03<00:52, 569.61it/s]","\rparsing log, completed traces ::   5%|4         | 1512/31509 [00:03<00:52, 569.58it/s]","\rparsing log, completed traces ::   5%|4         | 1570/31509 [00:03<00:52, 572.48it/s]","\rparsing log, completed traces ::   5%|5         | 1629/31509 [00:03<00:51, 575.75it/s]","\rparsing log, completed traces ::   5%|5         | 1687/31509 [00:03<00:53, 560.46it/s]","\rparsing log, completed traces ::   6%|5         | 1744/31509 [00:04<01:30, 330.37it/s]","\rparsing log, completed traces ::   6%|5         | 1792/31509 [00:04<01:22, 359.06it/s]","\rparsing log, completed traces ::   6%|5         | 1843/31509 [00:04<01:16, 389.76it/s]","\rparsing log, completed traces ::   6%|6         | 1901/31509 [00:04<01:08, 433.45it/s]","\rparsing log, completed traces ::   6%|6         | 1960/31509 [00:04<01:02, 472.80it/s]","\rparsing log, completed traces ::   6%|6         | 2018/31509 [00:04<00:59, 499.31it/s]","\rparsing log, completed traces ::   7%|6         | 2074/31509 [00:04<00:57, 515.68it/s]","\rparsing log, completed traces ::   7%|6         | 2133/31509 [00:04<00:54, 535.53it/s]","\rparsing log, completed traces ::   7%|6         | 2192/31509 [00:05<00:53, 548.07it/s]","\rparsing log, completed traces ::   7%|7         | 2252/31509 [00:05<00:52, 561.91it/s]","\rparsing log, completed traces ::   7%|7         | 2313/31509 [00:05<00:50, 573.28it/s]","\rparsing log, completed traces ::   8%|7         | 2372/31509 [00:05<00:51, 565.58it/s]","\rparsing log, completed traces ::   8%|7         | 2432/31509 [00:05<00:50, 575.22it/s]","\rparsing log, completed traces ::   8%|7         | 2492/31509 [00:05<00:50, 579.80it/s]","\rparsing log, completed traces ::   8%|8         | 2551/31509 [00:05<00:49, 582.54it/s]","\rparsing log, completed traces ::   8%|8         | 2618/31509 [00:05<00:47, 607.64it/s]","\rparsing log, completed traces ::   9%|8         | 2679/31509 [00:05<00:47, 603.10it/s]","\rparsing log, completed traces ::   9%|8         | 2740/31509 [00:06<01:35, 301.42it/s]","\rparsing log, completed traces ::   9%|8         | 2805/31509 [00:06<01:19, 361.20it/s]","\rparsing log, completed traces ::   9%|9         | 2873/31509 [00:06<01:07, 422.68it/s]","\rparsing log, completed traces ::   9%|9         | 2939/31509 [00:06<01:00, 473.37it/s]","\rparsing log, completed traces ::  10%|9         | 3007/31509 [00:06<00:54, 520.65it/s]","\rparsing log, completed traces ::  10%|9         | 3069/31509 [00:06<00:53, 530.50it/s]","\rparsing log, completed traces ::  10%|9         | 3129/31509 [00:06<00:52, 542.60it/s]","\rparsing log, completed traces ::  10%|#         | 3190/31509 [00:06<00:50, 559.77it/s]","\rparsing log, completed traces ::  10%|#         | 3250/31509 [00:07<00:50, 564.13it/s]","\rparsing log, completed traces ::  11%|#         | 3319/31509 [00:07<00:47, 598.07it/s]","\rparsing log, completed traces ::  11%|#         | 3381/31509 [00:07<00:46, 604.34it/s]","\rparsing log, completed traces ::  11%|#         | 3449/31509 [00:07<00:44, 624.66it/s]","\rparsing log, completed traces ::  11%|#1        | 3517/31509 [00:07<00:43, 639.57it/s]","\rparsing log, completed traces ::  11%|#1        | 3582/31509 [00:07<00:54, 515.35it/s]","\rparsing log, completed traces ::  12%|#1        | 3641/31509 [00:07<00:52, 533.11it/s]","\rparsing log, completed traces ::  12%|#1        | 3705/31509 [00:07<00:49, 560.08it/s]","\rparsing log, completed traces ::  12%|#1        | 3766/31509 [00:07<00:48, 571.98it/s]","\rparsing log, completed traces ::  12%|#2        | 3826/31509 [00:08<00:47, 576.90it/s]","\rparsing log, completed traces ::  12%|#2        | 3888/31509 [00:08<00:47, 586.29it/s]","\rparsing log, completed traces ::  13%|#2        | 3948/31509 [00:08<01:28, 309.89it/s]","\rparsing log, completed traces ::  13%|#2        | 4016/31509 [00:08<01:13, 374.60it/s]","\rparsing log, completed traces ::  13%|#2        | 4080/31509 [00:08<01:04, 427.12it/s]","\rparsing log, completed traces ::  13%|#3        | 4143/31509 [00:08<00:57, 471.95it/s]","\rparsing log, completed traces ::  13%|#3        | 4203/31509 [00:09<00:54, 501.89it/s]","\rparsing log, completed traces ::  14%|#3        | 4264/31509 [00:09<00:51, 526.18it/s]","\rparsing log, completed traces ::  14%|#3        | 4324/31509 [00:09<00:50, 543.37it/s]","\rparsing log, completed traces ::  14%|#3        | 4389/31509 [00:09<00:47, 569.35it/s]","\rparsing log, completed traces ::  14%|#4        | 4450/31509 [00:09<00:46, 575.89it/s]","\rparsing log, completed traces ::  14%|#4        | 4513/31509 [00:09<00:45, 590.74it/s]","\rparsing log, completed traces ::  15%|#4        | 4575/31509 [00:09<00:44, 598.86it/s]","\rparsing log, completed traces ::  15%|#4        | 4642/31509 [00:09<00:43, 618.63it/s]","\rparsing log, completed traces ::  15%|#4        | 4708/31509 [00:09<00:42, 630.41it/s]","\rparsing log, completed traces ::  15%|#5        | 4775/31509 [00:09<00:41, 640.73it/s]","\rparsing log, completed traces ::  15%|#5        | 4841/31509 [00:10<00:41, 643.61it/s]","\rparsing log, completed traces ::  16%|#5        | 4906/31509 [00:10<00:42, 630.23it/s]","\rparsing log, completed traces ::  16%|#5        | 4970/31509 [00:10<00:42, 631.71it/s]","\rparsing log, completed traces ::  16%|#5        | 5035/31509 [00:10<00:41, 636.23it/s]","\rparsing log, completed traces ::  16%|#6        | 5103/31509 [00:10<00:40, 648.34it/s]","\rparsing log, completed traces ::  16%|#6        | 5168/31509 [00:10<00:41, 633.99it/s]","\rparsing log, completed traces ::  17%|#6        | 5233/31509 [00:10<00:41, 637.86it/s]","\rparsing log, completed traces ::  17%|#6        | 5297/31509 [00:11<01:27, 300.06it/s]","\rparsing log, completed traces ::  17%|#7        | 5361/31509 [00:11<01:13, 355.54it/s]","\rparsing log, completed traces ::  17%|#7        | 5421/31509 [00:11<01:05, 401.25it/s]","\rparsing log, completed traces ::  17%|#7        | 5488/31509 [00:11<00:56, 458.43it/s]","\rparsing log, completed traces ::  18%|#7        | 5554/31509 [00:11<00:51, 504.80it/s]","\rparsing log, completed traces ::  18%|#7        | 5622/31509 [00:11<00:47, 548.04it/s]","\rparsing log, completed traces ::  18%|#8        | 5693/31509 [00:11<00:43, 588.75it/s]","\rparsing log, completed traces ::  18%|#8        | 5764/31509 [00:11<00:41, 620.12it/s]","\rparsing log, completed traces ::  19%|#8        | 5831/31509 [00:11<00:41, 622.28it/s]","\rparsing log, completed traces ::  19%|#8        | 5900/31509 [00:12<00:40, 639.28it/s]","\rparsing log, completed traces ::  19%|#8        | 5970/31509 [00:12<00:38, 656.16it/s]","\rparsing log, completed traces ::  19%|#9        | 6038/31509 [00:12<00:39, 644.31it/s]","\rparsing log, completed traces ::  19%|#9        | 6104/31509 [00:12<00:39, 636.75it/s]","\rparsing log, completed traces ::  20%|#9        | 6171/31509 [00:12<00:39, 645.89it/s]","\rparsing log, completed traces ::  20%|#9        | 6237/31509 [00:12<00:38, 649.59it/s]","\rparsing log, completed traces ::  20%|##        | 6303/31509 [00:12<00:39, 632.98it/s]","\rparsing log, completed traces ::  20%|##        | 6368/31509 [00:12<00:39, 637.30it/s]","\rparsing log, completed traces ::  20%|##        | 6433/31509 [00:12<00:39, 635.58it/s]","\rparsing log, completed traces ::  21%|##        | 6497/31509 [00:12<00:39, 634.68it/s]","\rparsing log, completed traces ::  21%|##        | 6561/31509 [00:13<00:41, 598.07it/s]","\rparsing log, completed traces ::  21%|##1       | 6622/31509 [00:13<00:41, 599.07it/s]","\rparsing log, completed traces ::  21%|##1       | 6686/31509 [00:13<00:40, 610.15it/s]","\rparsing log, completed traces ::  21%|##1       | 6749/31509 [00:13<00:40, 612.91it/s]","\rparsing log, completed traces ::  22%|##1       | 6811/31509 [00:13<01:27, 283.24it/s]","\rparsing log, completed traces ::  22%|##1       | 6873/31509 [00:13<01:13, 337.23it/s]","\rparsing log, completed traces ::  22%|##2       | 6940/31509 [00:14<01:01, 397.37it/s]","\rparsing log, completed traces ::  22%|##2       | 7001/31509 [00:14<00:55, 441.81it/s]","\rparsing log, completed traces ::  22%|##2       | 7072/31509 [00:14<00:48, 501.91it/s]","\rparsing log, completed traces ::  23%|##2       | 7138/31509 [00:14<00:45, 539.35it/s]","\rparsing log, completed traces ::  23%|##2       | 7201/31509 [00:14<00:44, 546.08it/s]","\rparsing log, completed traces ::  23%|##3       | 7266/31509 [00:14<00:42, 570.37it/s]","\rparsing log, completed traces ::  23%|##3       | 7328/31509 [00:14<00:41, 577.56it/s]","\rparsing log, completed traces ::  23%|##3       | 7390/31509 [00:14<00:41, 585.86it/s]","\rparsing log, completed traces ::  24%|##3       | 7451/31509 [00:14<00:40, 589.73it/s]","\rparsing log, completed traces ::  24%|##3       | 7522/31509 [00:15<00:38, 623.28it/s]","\rparsing log, completed traces ::  24%|##4       | 7588/31509 [00:15<00:37, 632.81it/s]","\rparsing log, completed traces ::  24%|##4       | 7653/31509 [00:15<00:38, 618.67it/s]","\rparsing log, completed traces ::  24%|##4       | 7716/31509 [00:15<00:38, 615.54it/s]","\rparsing log, completed traces ::  25%|##4       | 7779/31509 [00:15<00:39, 602.75it/s]","\rparsing log, completed traces ::  25%|##4       | 7845/31509 [00:15<00:38, 618.33it/s]","\rparsing log, completed traces ::  25%|##5       | 7909/31509 [00:15<00:37, 624.22it/s]","\rparsing log, completed traces ::  25%|##5       | 7974/31509 [00:15<00:37, 631.30it/s]","\rparsing log, completed traces ::  26%|##5       | 8038/31509 [00:15<00:37, 633.11it/s]","\rparsing log, completed traces ::  26%|##5       | 8102/31509 [00:15<00:36, 633.75it/s]","\rparsing log, completed traces ::  26%|##5       | 8166/31509 [00:16<00:37, 625.93it/s]","\rparsing log, completed traces ::  26%|##6       | 8236/31509 [00:16<00:36, 646.35it/s]","\rparsing log, completed traces ::  26%|##6       | 8301/31509 [00:16<00:36, 640.30it/s]","\rparsing log, completed traces ::  27%|##6       | 8366/31509 [00:16<00:36, 636.02it/s]","\rparsing log, completed traces ::  27%|##6       | 8430/31509 [00:16<00:37, 613.09it/s]","\rparsing log, completed traces ::  27%|##6       | 8496/31509 [00:17<01:26, 265.96it/s]","\rparsing log, completed traces ::  27%|##7       | 8551/31509 [00:17<01:14, 307.91it/s]","\rparsing log, completed traces ::  27%|##7       | 8613/31509 [00:17<01:03, 362.38it/s]","\rparsing log, completed traces ::  28%|##7       | 8676/31509 [00:17<00:55, 414.23it/s]","\rparsing log, completed traces ::  28%|##7       | 8738/31509 [00:17<00:49, 459.62it/s]","\rparsing log, completed traces ::  28%|##7       | 8801/31509 [00:17<00:45, 498.65it/s]","\rparsing log, completed traces ::  28%|##8       | 8862/31509 [00:17<00:43, 526.22it/s]","\rparsing log, completed traces ::  28%|##8       | 8922/31509 [00:17<00:41, 540.40it/s]","\rparsing log, completed traces ::  29%|##8       | 8984/31509 [00:17<00:40, 560.37it/s]","\rparsing log, completed traces ::  29%|##8       | 9044/31509 [00:17<00:40, 549.77it/s]","\rparsing log, completed traces ::  29%|##8       | 9105/31509 [00:18<00:39, 566.23it/s]","\rparsing log, completed traces ::  29%|##9       | 9168/31509 [00:18<00:38, 581.00it/s]","\rparsing log, completed traces ::  29%|##9       | 9228/31509 [00:18<00:38, 573.06it/s]","\rparsing log, completed traces ::  29%|##9       | 9287/31509 [00:18<00:38, 572.85it/s]","\rparsing log, completed traces ::  30%|##9       | 9352/31509 [00:18<00:37, 594.58it/s]","\rparsing log, completed traces ::  30%|##9       | 9413/31509 [00:18<00:37, 586.48it/s]","\rparsing log, completed traces ::  30%|###       | 9475/31509 [00:18<00:37, 591.99it/s]","\rparsing log, completed traces ::  30%|###       | 9537/31509 [00:18<00:36, 598.37it/s]","\rparsing log, completed traces ::  30%|###       | 9598/31509 [00:18<00:37, 592.07it/s]","\rparsing log, completed traces ::  31%|###       | 9659/31509 [00:18<00:36, 596.83it/s]","\rparsing log, completed traces ::  31%|###       | 9719/31509 [00:19<00:37, 584.14it/s]","\rparsing log, completed traces ::  31%|###1      | 9782/31509 [00:19<00:36, 596.51it/s]","\rparsing log, completed traces ::  31%|###1      | 9842/31509 [00:19<00:36, 592.65it/s]","\rparsing log, completed traces ::  31%|###1      | 9902/31509 [00:19<00:36, 586.26it/s]","\rparsing log, completed traces ::  32%|###1      | 9962/31509 [00:19<00:36, 587.53it/s]","\rparsing log, completed traces ::  32%|###1      | 10021/31509 [00:19<00:36, 581.01it/s]","\rparsing log, completed traces ::  32%|###1      | 10080/31509 [00:19<00:38, 562.77it/s]","\rparsing log, completed traces ::  32%|###2      | 10137/31509 [00:19<00:38, 554.09it/s]","\rparsing log, completed traces ::  32%|###2      | 10198/31509 [00:19<00:37, 570.06it/s]","\rparsing log, completed traces ::  33%|###2      | 10262/31509 [00:20<00:36, 586.61it/s]","\rparsing log, completed traces ::  33%|###2      | 10321/31509 [00:20<01:29, 236.33it/s]","\rparsing log, completed traces ::  33%|###2      | 10377/31509 [00:20<01:14, 282.81it/s]","\rparsing log, completed traces ::  33%|###3      | 10437/31509 [00:20<01:02, 336.24it/s]","\rparsing log, completed traces ::  33%|###3      | 10492/31509 [00:20<00:55, 377.53it/s]","\rparsing log, completed traces ::  33%|###3      | 10550/31509 [00:21<00:49, 421.23it/s]","\rparsing log, completed traces ::  34%|###3      | 10609/31509 [00:21<00:45, 460.67it/s]","\rparsing log, completed traces ::  34%|###3      | 10668/31509 [00:21<00:42, 492.37it/s]","\rparsing log, completed traces ::  34%|###4      | 10725/31509 [00:21<00:40, 508.04it/s]","\rparsing log, completed traces ::  34%|###4      | 10781/31509 [00:21<00:39, 518.57it/s]","\rparsing log, completed traces ::  34%|###4      | 10837/31509 [00:21<00:39, 520.27it/s]","\rparsing log, completed traces ::  35%|###4      | 10892/31509 [00:21<00:39, 520.35it/s]","\rparsing log, completed traces ::  35%|###4      | 10949/31509 [00:21<00:38, 533.18it/s]","\rparsing log, completed traces ::  35%|###4      | 11009/31509 [00:21<00:37, 551.38it/s]","\rparsing log, completed traces ::  35%|###5      | 11070/31509 [00:21<00:35, 568.06it/s]","\rparsing log, completed traces ::  35%|###5      | 11133/31509 [00:22<00:34, 583.99it/s]","\rparsing log, completed traces ::  36%|###5      | 11192/31509 [00:22<00:34, 585.67it/s]","\rparsing log, completed traces ::  36%|###5      | 11257/31509 [00:22<00:33, 602.47it/s]","\rparsing log, completed traces ::  36%|###5      | 11320/31509 [00:22<00:33, 607.74it/s]","\rparsing log, completed traces ::  36%|###6      | 11381/31509 [00:22<00:34, 585.74it/s]","\rparsing log, completed traces ::  36%|###6      | 11442/31509 [00:22<00:33, 591.10it/s]","\rparsing log, completed traces ::  37%|###6      | 11502/31509 [00:22<00:34, 583.51it/s]","\rparsing log, completed traces ::  37%|###6      | 11561/31509 [00:22<00:34, 577.26it/s]","\rparsing log, completed traces ::  37%|###6      | 11619/31509 [00:22<00:34, 577.39it/s]","\rparsing log, completed traces ::  37%|###7      | 11677/31509 [00:23<00:34, 574.34it/s]","\rparsing log, completed traces ::  37%|###7      | 11738/31509 [00:23<00:33, 583.22it/s]","\rparsing log, completed traces ::  37%|###7      | 11798/31509 [00:23<00:33, 584.85it/s]","\rparsing log, completed traces ::  38%|###7      | 11858/31509 [00:23<00:33, 587.44it/s]","\rparsing log, completed traces ::  38%|###7      | 11925/31509 [00:23<00:32, 611.46it/s]","\rparsing log, completed traces ::  38%|###8      | 11987/31509 [00:23<00:31, 613.98it/s]","\rparsing log, completed traces ::  38%|###8      | 12049/31509 [00:23<00:31, 614.65it/s]","\rparsing log, completed traces ::  38%|###8      | 12111/31509 [00:23<00:32, 601.98it/s]","\rparsing log, completed traces ::  39%|###8      | 12172/31509 [00:23<00:32, 596.30it/s]","\rparsing log, completed traces ::  39%|###8      | 12238/31509 [00:23<00:31, 605.21it/s]","\rparsing log, completed traces ::  39%|###9      | 12299/31509 [00:24<01:27, 219.60it/s]","\rparsing log, completed traces ::  39%|###9      | 12362/31509 [00:24<01:10, 272.73it/s]","\rparsing log, completed traces ::  39%|###9      | 12425/31509 [00:24<00:58, 328.42it/s]","\rparsing log, completed traces ::  40%|###9      | 12486/31509 [00:24<00:50, 378.47it/s]","\rparsing log, completed traces ::  40%|###9      | 12544/31509 [00:25<00:45, 419.39it/s]","\rparsing log, completed traces ::  40%|####      | 12607/31509 [00:25<00:40, 466.87it/s]","\rparsing log, completed traces ::  40%|####      | 12667/31509 [00:25<00:37, 498.86it/s]","\rparsing log, completed traces ::  40%|####      | 12733/31509 [00:25<00:34, 538.85it/s]","\rparsing log, completed traces ::  41%|####      | 12795/31509 [00:25<00:33, 559.97it/s]","\rparsing log, completed traces ::  41%|####      | 12856/31509 [00:25<00:33, 563.60it/s]","\rparsing log, completed traces ::  41%|####      | 12916/31509 [00:25<00:32, 565.46it/s]","\rparsing log, completed traces ::  41%|####1     | 12981/31509 [00:25<00:31, 587.87it/s]","\rparsing log, completed traces ::  41%|####1     | 13042/31509 [00:25<00:31, 593.86it/s]","\rparsing log, completed traces ::  42%|####1     | 13112/31509 [00:25<00:29, 621.11it/s]","\rparsing log, completed traces ::  42%|####1     | 13176/31509 [00:26<00:30, 606.28it/s]","\rparsing log, completed traces ::  42%|####2     | 13238/31509 [00:26<00:30, 605.70it/s]","\rparsing log, completed traces ::  42%|####2     | 13300/31509 [00:26<00:30, 600.54it/s]","\rparsing log, completed traces ::  42%|####2     | 13361/31509 [00:26<00:30, 596.90it/s]","\rparsing log, completed traces ::  43%|####2     | 13424/31509 [00:26<00:29, 603.79it/s]","\rparsing log, completed traces ::  43%|####2     | 13485/31509 [00:26<00:29, 605.19it/s]","\rparsing log, completed traces ::  43%|####2     | 13546/31509 [00:26<00:29, 601.38it/s]","\rparsing log, completed traces ::  43%|####3     | 13607/31509 [00:26<00:29, 598.28it/s]","\rparsing log, completed traces ::  43%|####3     | 13667/31509 [00:26<00:30, 586.59it/s]","\rparsing log, completed traces ::  44%|####3     | 13731/31509 [00:26<00:29, 601.34it/s]","\rparsing log, completed traces ::  44%|####3     | 13792/31509 [00:27<00:29, 599.42it/s]","\rparsing log, completed traces ::  44%|####3     | 13854/31509 [00:27<00:29, 599.42it/s]","\rparsing log, completed traces ::  44%|####4     | 13914/31509 [00:27<00:29, 598.46it/s]","\rparsing log, completed traces ::  44%|####4     | 13975/31509 [00:27<00:29, 599.57it/s]","\rparsing log, completed traces ::  45%|####4     | 14040/31509 [00:27<00:28, 612.86it/s]","\rparsing log, completed traces ::  45%|####4     | 14102/31509 [00:27<00:28, 601.45it/s]","\rparsing log, completed traces ::  45%|####4     | 14169/31509 [00:27<00:28, 617.80it/s]","\rparsing log, completed traces ::  45%|####5     | 14231/31509 [00:27<00:28, 617.00it/s]","\rparsing log, completed traces ::  45%|####5     | 14295/31509 [00:27<00:27, 622.01it/s]","\rparsing log, completed traces ::  46%|####5     | 14361/31509 [00:28<00:27, 632.05it/s]","\rparsing log, completed traces ::  46%|####5     | 14428/31509 [00:28<00:26, 643.02it/s]","\rparsing log, completed traces ::  46%|####6     | 14495/31509 [00:28<00:26, 646.41it/s]","\rparsing log, completed traces ::  46%|####6     | 14560/31509 [00:28<00:27, 623.82it/s]","\rparsing log, completed traces ::  46%|####6     | 14623/31509 [00:29<01:20, 210.69it/s]","\rparsing log, completed traces ::  47%|####6     | 14683/31509 [00:29<01:05, 258.11it/s]","\rparsing log, completed traces ::  47%|####6     | 14743/31509 [00:29<00:54, 308.73it/s]","\rparsing log, completed traces ::  47%|####6     | 14809/31509 [00:29<00:45, 369.36it/s]","\rparsing log, completed traces ::  47%|####7     | 14868/31509 [00:29<00:40, 411.14it/s]","\rparsing log, completed traces ::  47%|####7     | 14932/31509 [00:29<00:35, 461.32it/s]","\rparsing log, completed traces ::  48%|####7     | 14993/31509 [00:29<00:33, 495.92it/s]","\rparsing log, completed traces ::  48%|####7     | 15056/31509 [00:29<00:31, 529.87it/s]","\rparsing log, completed traces ::  48%|####7     | 15118/31509 [00:29<00:29, 549.55it/s]","\rparsing log, completed traces ::  48%|####8     | 15183/31509 [00:30<00:28, 576.58it/s]","\rparsing log, completed traces ::  48%|####8     | 15245/31509 [00:30<00:27, 586.18it/s]","\rparsing log, completed traces ::  49%|####8     | 15313/31509 [00:30<00:26, 609.43it/s]","\rparsing log, completed traces ::  49%|####8     | 15377/31509 [00:30<00:26, 616.63it/s]","\rparsing log, completed traces ::  49%|####9     | 15443/31509 [00:30<00:25, 628.07it/s]","\rparsing log, completed traces ::  49%|####9     | 15507/31509 [00:30<00:25, 623.60it/s]","\rparsing log, completed traces ::  49%|####9     | 15576/31509 [00:30<00:24, 641.33it/s]","\rparsing log, completed traces ::  50%|####9     | 15641/31509 [00:30<00:24, 636.58it/s]","\rparsing log, completed traces ::  50%|####9     | 15706/31509 [00:30<00:24, 635.35it/s]","\rparsing log, completed traces ::  50%|#####     | 15770/31509 [00:30<00:25, 618.92it/s]","\rparsing log, completed traces ::  50%|#####     | 15833/31509 [00:31<00:25, 614.44it/s]","\rparsing log, completed traces ::  50%|#####     | 15900/31509 [00:31<00:24, 629.17it/s]","\rparsing log, completed traces ::  51%|#####     | 15964/31509 [00:31<00:24, 631.22it/s]","\rparsing log, completed traces ::  51%|#####     | 16028/31509 [00:31<00:25, 619.05it/s]","\rparsing log, completed traces ::  51%|#####1    | 16094/31509 [00:31<00:24, 630.23it/s]","\rparsing log, completed traces ::  51%|#####1    | 16162/31509 [00:31<00:23, 644.19it/s]","\rparsing log, completed traces ::  52%|#####1    | 16229/31509 [00:31<00:23, 651.11it/s]","\rparsing log, completed traces ::  52%|#####1    | 16295/31509 [00:31<00:23, 648.99it/s]","\rparsing log, completed traces ::  52%|#####1    | 16360/31509 [00:31<00:24, 625.32it/s]","\rparsing log, completed traces ::  52%|#####2    | 16423/31509 [00:31<00:24, 609.86it/s]","\rparsing log, completed traces ::  52%|#####2    | 16485/31509 [00:32<00:24, 603.78it/s]","\rparsing log, completed traces ::  53%|#####2    | 16546/31509 [00:32<00:25, 596.01it/s]","\rparsing log, completed traces ::  53%|#####2    | 16608/31509 [00:32<00:24, 601.47it/s]","\rparsing log, completed traces ::  53%|#####2    | 16674/31509 [00:32<00:24, 614.28it/s]","\rparsing log, completed traces ::  53%|#####3    | 16736/31509 [00:32<00:24, 610.10it/s]","\rparsing log, completed traces ::  53%|#####3    | 16798/31509 [00:32<00:24, 611.26it/s]","\rparsing log, completed traces ::  54%|#####3    | 16861/31509 [00:32<00:23, 614.92it/s]","\rparsing log, completed traces ::  54%|#####3    | 16926/31509 [00:32<00:23, 622.78it/s]","\rparsing log, completed traces ::  54%|#####3    | 16989/31509 [00:32<00:23, 618.95it/s]","\rparsing log, completed traces ::  54%|#####4    | 17060/31509 [00:33<00:22, 642.90it/s]","\rparsing log, completed traces ::  54%|#####4    | 17125/31509 [00:33<00:23, 621.03it/s]","\rparsing log, completed traces ::  55%|#####4    | 17188/31509 [00:33<00:22, 623.17it/s]","\rparsing log, completed traces ::  55%|#####4    | 17251/31509 [00:33<00:22, 621.40it/s]","\rparsing log, completed traces ::  55%|#####4    | 17316/31509 [00:33<00:22, 628.45it/s]","\rparsing log, completed traces ::  55%|#####5    | 17379/31509 [00:34<01:09, 202.13it/s]","\rparsing log, completed traces ::  55%|#####5    | 17450/31509 [00:34<00:53, 262.31it/s]","\rparsing log, completed traces ::  56%|#####5    | 17506/31509 [00:34<00:45, 305.49it/s]","\rparsing log, completed traces ::  56%|#####5    | 17579/31509 [00:34<00:36, 377.23it/s]","\rparsing log, completed traces ::  56%|#####6    | 17653/31509 [00:34<00:30, 448.35it/s]","\rparsing log, completed traces ::  56%|#####6    | 17722/31509 [00:34<00:27, 500.27it/s]","\rparsing log, completed traces ::  56%|#####6    | 17792/31509 [00:34<00:25, 545.31it/s]","\rparsing log, completed traces ::  57%|#####6    | 17859/31509 [00:34<00:23, 574.95it/s]","\rparsing log, completed traces ::  57%|#####6    | 17926/31509 [00:35<00:23, 589.08it/s]","\rparsing log, completed traces ::  57%|#####7    | 17994/31509 [00:35<00:22, 612.46it/s]","\rparsing log, completed traces ::  57%|#####7    | 18060/31509 [00:35<00:21, 624.70it/s]","\rparsing log, completed traces ::  58%|#####7    | 18127/31509 [00:35<00:21, 636.25it/s]","\rparsing log, completed traces ::  58%|#####7    | 18196/31509 [00:35<00:20, 649.61it/s]","\rparsing log, completed traces ::  58%|#####7    | 18263/31509 [00:35<00:20, 646.86it/s]","\rparsing log, completed traces ::  58%|#####8    | 18329/31509 [00:35<00:21, 626.79it/s]","\rparsing log, completed traces ::  58%|#####8    | 18397/31509 [00:35<00:20, 639.93it/s]","\rparsing log, completed traces ::  59%|#####8    | 18462/31509 [00:35<00:20, 635.45it/s]","\rparsing log, completed traces ::  59%|#####8    | 18527/31509 [00:35<00:20, 620.99it/s]","\rparsing log, completed traces ::  59%|#####8    | 18590/31509 [00:36<00:21, 613.70it/s]","\rparsing log, completed traces ::  59%|#####9    | 18661/31509 [00:36<00:20, 639.06it/s]","\rparsing log, completed traces ::  59%|#####9    | 18730/31509 [00:36<00:19, 650.54it/s]","\rparsing log, completed traces ::  60%|#####9    | 18796/31509 [00:36<00:19, 641.12it/s]","\rparsing log, completed traces ::  60%|#####9    | 18861/31509 [00:36<00:19, 639.30it/s]","\rparsing log, completed traces ::  60%|######    | 18934/31509 [00:36<00:18, 665.49it/s]","\rparsing log, completed traces ::  60%|######    | 19001/31509 [00:36<00:19, 649.71it/s]","\rparsing log, completed traces ::  61%|######    | 19067/31509 [00:36<00:19, 646.25it/s]","\rparsing log, completed traces ::  61%|######    | 19133/31509 [00:36<00:19, 650.07it/s]","\rparsing log, completed traces ::  61%|######    | 19199/31509 [00:37<00:19, 643.17it/s]","\rparsing log, completed traces ::  61%|######1   | 19264/31509 [00:37<00:19, 643.61it/s]","\rparsing log, completed traces ::  61%|######1   | 19329/31509 [00:37<00:18, 642.16it/s]","\rparsing log, completed traces ::  62%|######1   | 19398/31509 [00:37<00:18, 656.02it/s]","\rparsing log, completed traces ::  62%|######1   | 19464/31509 [00:37<00:18, 655.06it/s]","\rparsing log, completed traces ::  62%|######1   | 19530/31509 [00:37<00:18, 646.03it/s]","\rparsing log, completed traces ::  62%|######2   | 19596/31509 [00:37<00:18, 646.08it/s]","\rparsing log, completed traces ::  62%|######2   | 19661/31509 [00:37<00:18, 646.13it/s]","\rparsing log, completed traces ::  63%|######2   | 19726/31509 [00:37<00:18, 646.76it/s]","\rparsing log, completed traces ::  63%|######2   | 19791/31509 [00:37<00:18, 626.05it/s]","\rparsing log, completed traces ::  63%|######3   | 19854/31509 [00:38<00:18, 616.09it/s]","\rparsing log, completed traces ::  63%|######3   | 19919/31509 [00:38<00:18, 625.42it/s]","\rparsing log, completed traces ::  63%|######3   | 19982/31509 [00:38<00:18, 619.95it/s]","\rparsing log, completed traces ::  64%|######3   | 20048/31509 [00:38<00:18, 628.15it/s]","\rparsing log, completed traces ::  64%|######3   | 20111/31509 [00:38<00:18, 626.07it/s]","\rparsing log, completed traces ::  64%|######4   | 20177/31509 [00:38<00:17, 633.13it/s]","\rparsing log, completed traces ::  64%|######4   | 20244/31509 [00:38<00:17, 643.25it/s]","\rparsing log, completed traces ::  64%|######4   | 20312/31509 [00:38<00:17, 649.58it/s]","\rparsing log, completed traces ::  65%|######4   | 20377/31509 [00:38<00:17, 642.42it/s]","\rparsing log, completed traces ::  65%|######4   | 20451/31509 [00:38<00:16, 670.42it/s]","\rparsing log, completed traces ::  65%|######5   | 20519/31509 [00:39<00:55, 197.71it/s]","\rparsing log, completed traces ::  65%|######5   | 20587/31509 [00:39<00:43, 250.64it/s]","\rparsing log, completed traces ::  66%|######5   | 20659/31509 [00:40<00:34, 314.49it/s]","\rparsing log, completed traces ::  66%|######5   | 20727/31509 [00:40<00:28, 373.98it/s]","\rparsing log, completed traces ::  66%|######5   | 20792/31509 [00:40<00:25, 425.49it/s]","\rparsing log, completed traces ::  66%|######6   | 20859/31509 [00:40<00:22, 475.99it/s]","\rparsing log, completed traces ::  66%|######6   | 20923/31509 [00:40<00:20, 506.87it/s]","\rparsing log, completed traces ::  67%|######6   | 20986/31509 [00:40<00:19, 537.11it/s]","\rparsing log, completed traces ::  67%|######6   | 21049/31509 [00:40<00:18, 557.39it/s]","\rparsing log, completed traces ::  67%|######7   | 21112/31509 [00:40<00:18, 575.81it/s]","\rparsing log, completed traces ::  67%|######7   | 21175/31509 [00:40<00:17, 586.09it/s]","\rparsing log, completed traces ::  67%|######7   | 21244/31509 [00:41<00:16, 613.77it/s]","\rparsing log, completed traces ::  68%|######7   | 21310/31509 [00:41<00:16, 626.39it/s]","\rparsing log, completed traces ::  68%|######7   | 21375/31509 [00:41<00:16, 623.94it/s]","\rparsing log, completed traces ::  68%|######8   | 21445/31509 [00:41<00:15, 645.32it/s]","\rparsing log, completed traces ::  68%|######8   | 21515/31509 [00:41<00:15, 659.42it/s]","\rparsing log, completed traces ::  69%|######8   | 21586/31509 [00:41<00:14, 673.10it/s]","\rparsing log, completed traces ::  69%|######8   | 21654/31509 [00:41<00:15, 649.79it/s]","\rparsing log, completed traces ::  69%|######8   | 21723/31509 [00:41<00:14, 658.41it/s]","\rparsing log, completed traces ::  69%|######9   | 21793/31509 [00:41<00:14, 669.58it/s]","\rparsing log, completed traces ::  69%|######9   | 21865/31509 [00:41<00:14, 683.36it/s]","\rparsing log, completed traces ::  70%|######9   | 21934/31509 [00:42<00:14, 680.40it/s]","\rparsing log, completed traces ::  70%|######9   | 22003/31509 [00:42<00:14, 672.63it/s]","\rparsing log, completed traces ::  70%|#######   | 22076/31509 [00:42<00:13, 686.59it/s]","\rparsing log, completed traces ::  70%|#######   | 22145/31509 [00:42<00:13, 684.52it/s]","\rparsing log, completed traces ::  71%|#######   | 22214/31509 [00:42<00:13, 678.54it/s]","\rparsing log, completed traces ::  71%|#######   | 22283/31509 [00:42<00:13, 679.90it/s]","\rparsing log, completed traces ::  71%|#######   | 22352/31509 [00:42<00:13, 662.08it/s]","\rparsing log, completed traces ::  71%|#######1  | 22421/31509 [00:42<00:13, 667.97it/s]","\rparsing log, completed traces ::  71%|#######1  | 22491/31509 [00:42<00:13, 674.23it/s]","\rparsing log, completed traces ::  72%|#######1  | 22559/31509 [00:42<00:13, 661.61it/s]","\rparsing log, completed traces ::  72%|#######1  | 22627/31509 [00:43<00:13, 664.02it/s]","\rparsing log, completed traces ::  72%|#######2  | 22694/31509 [00:43<00:13, 663.66it/s]","\rparsing log, completed traces ::  72%|#######2  | 22762/31509 [00:43<00:13, 668.13it/s]","\rparsing log, completed traces ::  72%|#######2  | 22829/31509 [00:43<00:13, 667.63it/s]","\rparsing log, completed traces ::  73%|#######2  | 22896/31509 [00:43<00:12, 664.91it/s]","\rparsing log, completed traces ::  73%|#######2  | 22963/31509 [00:43<00:13, 647.54it/s]","\rparsing log, completed traces ::  73%|#######3  | 23028/31509 [00:43<00:13, 646.42it/s]","\rparsing log, completed traces ::  73%|#######3  | 23093/31509 [00:43<00:13, 642.90it/s]","\rparsing log, completed traces ::  73%|#######3  | 23158/31509 [00:43<00:13, 633.57it/s]","\rparsing log, completed traces ::  74%|#######3  | 23226/31509 [00:43<00:12, 644.53it/s]","\rparsing log, completed traces ::  74%|#######3  | 23291/31509 [00:44<00:12, 634.83it/s]","\rparsing log, completed traces ::  74%|#######4  | 23355/31509 [00:44<00:13, 622.12it/s]","\rparsing log, completed traces ::  74%|#######4  | 23420/31509 [00:44<00:12, 626.55it/s]","\rparsing log, completed traces ::  75%|#######4  | 23484/31509 [00:44<00:12, 627.58it/s]","\rparsing log, completed traces ::  75%|#######4  | 23550/31509 [00:44<00:12, 635.28it/s]","\rparsing log, completed traces ::  75%|#######4  | 23618/31509 [00:44<00:12, 645.07it/s]","\rparsing log, completed traces ::  75%|#######5  | 23683/31509 [00:44<00:12, 641.90it/s]","\rparsing log, completed traces ::  75%|#######5  | 23749/31509 [00:44<00:12, 646.40it/s]","\rparsing log, completed traces ::  76%|#######5  | 23814/31509 [00:44<00:11, 646.53it/s]","\rparsing log, completed traces ::  76%|#######5  | 23879/31509 [00:45<00:11, 645.01it/s]","\rparsing log, completed traces ::  76%|#######5  | 23944/31509 [00:45<00:11, 637.90it/s]","\rparsing log, completed traces ::  76%|#######6  | 24008/31509 [00:46<00:42, 176.50it/s]","\rparsing log, completed traces ::  76%|#######6  | 24072/31509 [00:46<00:33, 224.86it/s]","\rparsing log, completed traces ::  77%|#######6  | 24136/31509 [00:46<00:26, 278.36it/s]","\rparsing log, completed traces ::  77%|#######6  | 24199/31509 [00:46<00:21, 332.67it/s]","\rparsing log, completed traces ::  77%|#######7  | 24264/31509 [00:46<00:18, 390.18it/s]","\rparsing log, completed traces ::  77%|#######7  | 24332/31509 [00:46<00:15, 448.57it/s]","\rparsing log, completed traces ::  77%|#######7  | 24396/31509 [00:46<00:14, 491.99it/s]","\rparsing log, completed traces ::  78%|#######7  | 24459/31509 [00:46<00:13, 523.01it/s]","\rparsing log, completed traces ::  78%|#######7  | 24527/31509 [00:46<00:12, 561.50it/s]","\rparsing log, completed traces ::  78%|#######8  | 24597/31509 [00:47<00:11, 597.77it/s]","\rparsing log, completed traces ::  78%|#######8  | 24664/31509 [00:47<00:11, 617.74it/s]","\rparsing log, completed traces ::  78%|#######8  | 24730/31509 [00:47<00:10, 628.99it/s]","\rparsing log, completed traces ::  79%|#######8  | 24797/31509 [00:47<00:10, 638.73it/s]","\rparsing log, completed traces ::  79%|#######8  | 24866/31509 [00:47<00:10, 652.50it/s]","\rparsing log, completed traces ::  79%|#######9  | 24933/31509 [00:47<00:10, 649.12it/s]","\rparsing log, completed traces ::  79%|#######9  | 25006/31509 [00:47<00:09, 670.87it/s]","\rparsing log, completed traces ::  80%|#######9  | 25074/31509 [00:47<00:10, 637.60it/s]","\rparsing log, completed traces ::  80%|#######9  | 25141/31509 [00:47<00:09, 642.61it/s]","\rparsing log, completed traces ::  80%|########  | 25211/31509 [00:47<00:09, 658.40it/s]","\rparsing log, completed traces ::  80%|########  | 25278/31509 [00:48<00:09, 638.75it/s]","\rparsing log, completed traces ::  80%|########  | 25343/31509 [00:48<00:09, 620.19it/s]","\rparsing log, completed traces ::  81%|########  | 25416/31509 [00:48<00:09, 649.55it/s]","\rparsing log, completed traces ::  81%|########  | 25482/31509 [00:48<00:09, 647.74it/s]","\rparsing log, completed traces ::  81%|########1 | 25548/31509 [00:48<00:09, 644.67it/s]","\rparsing log, completed traces ::  81%|########1 | 25614/31509 [00:48<00:09, 648.54it/s]","\rparsing log, completed traces ::  82%|########1 | 25680/31509 [00:48<00:09, 639.81it/s]","\rparsing log, completed traces ::  82%|########1 | 25745/31509 [00:48<00:08, 642.09it/s]","\rparsing log, completed traces ::  82%|########1 | 25810/31509 [00:48<00:08, 636.04it/s]","\rparsing log, completed traces ::  82%|########2 | 25874/31509 [00:48<00:08, 632.45it/s]","\rparsing log, completed traces ::  82%|########2 | 25940/31509 [00:49<00:08, 636.80it/s]","\rparsing log, completed traces ::  83%|########2 | 26004/31509 [00:49<00:08, 622.28it/s]","\rparsing log, completed traces ::  83%|########2 | 26067/31509 [00:49<00:08, 613.53it/s]","\rparsing log, completed traces ::  83%|########2 | 26131/31509 [00:49<00:08, 619.98it/s]","\rparsing log, completed traces ::  83%|########3 | 26196/31509 [00:49<00:08, 628.18it/s]","\rparsing log, completed traces ::  83%|########3 | 26259/31509 [00:49<00:08, 624.59it/s]","\rparsing log, completed traces ::  84%|########3 | 26324/31509 [00:49<00:08, 631.28it/s]","\rparsing log, completed traces ::  84%|########3 | 26391/31509 [00:49<00:08, 638.30it/s]","\rparsing log, completed traces ::  84%|########3 | 26455/31509 [00:49<00:07, 638.58it/s]","\rparsing log, completed traces ::  84%|########4 | 26519/31509 [00:50<00:07, 627.69it/s]","\rparsing log, completed traces ::  84%|########4 | 26582/31509 [00:50<00:08, 615.27it/s]","\rparsing log, completed traces ::  85%|########4 | 26647/31509 [00:50<00:07, 621.90it/s]","\rparsing log, completed traces ::  85%|########4 | 26711/31509 [00:50<00:07, 624.94it/s]","\rparsing log, completed traces ::  85%|########4 | 26775/31509 [00:50<00:07, 628.14it/s]","\rparsing log, completed traces ::  85%|########5 | 26839/31509 [00:50<00:07, 629.91it/s]","\rparsing log, completed traces ::  85%|########5 | 26903/31509 [00:50<00:07, 614.89it/s]","\rparsing log, completed traces ::  86%|########5 | 26965/31509 [00:50<00:07, 613.69it/s]","\rparsing log, completed traces ::  86%|########5 | 27027/31509 [00:50<00:07, 604.15it/s]","\rparsing log, completed traces ::  86%|########5 | 27089/31509 [00:50<00:07, 605.76it/s]","\rparsing log, completed traces ::  86%|########6 | 27157/31509 [00:51<00:06, 626.93it/s]","\rparsing log, completed traces ::  86%|########6 | 27223/31509 [00:51<00:06, 633.96it/s]","\rparsing log, completed traces ::  87%|########6 | 27289/31509 [00:51<00:06, 641.64it/s]","\rparsing log, completed traces ::  87%|########6 | 27354/31509 [00:51<00:06, 618.44it/s]","\rparsing log, completed traces ::  87%|########7 | 27420/31509 [00:51<00:06, 630.27it/s]","\rparsing log, completed traces ::  87%|########7 | 27484/31509 [00:51<00:06, 630.50it/s]","\rparsing log, completed traces ::  87%|########7 | 27548/31509 [00:51<00:06, 627.16it/s]","\rparsing log, completed traces ::  88%|########7 | 27615/31509 [00:51<00:06, 639.19it/s]","\rparsing log, completed traces ::  88%|########7 | 27682/31509 [00:51<00:05, 647.81it/s]","\rparsing log, completed traces ::  88%|########8 | 27751/31509 [00:51<00:05, 658.19it/s]","\rparsing log, completed traces ::  88%|########8 | 27817/31509 [00:52<00:05, 647.03it/s]","\rparsing log, completed traces ::  88%|########8 | 27882/31509 [00:53<00:22, 160.99it/s]","\rparsing log, completed traces ::  89%|########8 | 27945/31509 [00:53<00:17, 205.07it/s]","\rparsing log, completed traces ::  89%|########8 | 28008/31509 [00:53<00:13, 255.05it/s]","\rparsing log, completed traces ::  89%|########9 | 28070/31509 [00:53<00:11, 306.99it/s]","\rparsing log, completed traces ::  89%|########9 | 28131/31509 [00:53<00:09, 357.72it/s]","\rparsing log, completed traces ::  89%|########9 | 28193/31509 [00:53<00:08, 409.01it/s]","\rparsing log, completed traces ::  90%|########9 | 28260/31509 [00:53<00:06, 465.02it/s]","\rparsing log, completed traces ::  90%|########9 | 28322/31509 [00:53<00:06, 496.77it/s]","\rparsing log, completed traces ::  90%|######### | 28388/31509 [00:54<00:05, 537.76it/s]","\rparsing log, completed traces ::  90%|######### | 28451/31509 [00:54<00:05, 550.77it/s]","\rparsing log, completed traces ::  91%|######### | 28521/31509 [00:54<00:05, 588.39it/s]","\rparsing log, completed traces ::  91%|######### | 28590/31509 [00:54<00:04, 613.94it/s]","\rparsing log, completed traces ::  91%|######### | 28655/31509 [00:54<00:04, 606.80it/s]","\rparsing log, completed traces ::  91%|#########1| 28719/31509 [00:54<00:04, 598.22it/s]","\rparsing log, completed traces ::  91%|#########1| 28781/31509 [00:54<00:04, 582.06it/s]","\rparsing log, completed traces ::  92%|#########1| 28841/31509 [00:54<00:04, 578.93it/s]","\rparsing log, completed traces ::  92%|#########1| 28900/31509 [00:54<00:04, 576.40it/s]","\rparsing log, completed traces ::  92%|#########1| 28963/31509 [00:54<00:04, 590.54it/s]","\rparsing log, completed traces ::  92%|#########2| 29027/31509 [00:55<00:04, 604.53it/s]","\rparsing log, completed traces ::  92%|#########2| 29088/31509 [00:55<00:04, 595.61it/s]","\rparsing log, completed traces ::  93%|#########2| 29149/31509 [00:55<00:03, 599.42it/s]","\rparsing log, completed traces ::  93%|#########2| 29216/31509 [00:55<00:03, 618.15it/s]","\rparsing log, completed traces ::  93%|#########2| 29278/31509 [00:55<00:03, 616.71it/s]","\rparsing log, completed traces ::  93%|#########3| 29341/31509 [00:55<00:03, 619.41it/s]","\rparsing log, completed traces ::  93%|#########3| 29404/31509 [00:55<00:03, 608.26it/s]","\rparsing log, completed traces ::  94%|#########3| 29465/31509 [00:55<00:03, 594.07it/s]","\rparsing log, completed traces ::  94%|#########3| 29525/31509 [00:55<00:03, 576.29it/s]","\rparsing log, completed traces ::  94%|#########3| 29585/31509 [00:56<00:03, 580.68it/s]","\rparsing log, completed traces ::  94%|#########4| 29653/31509 [00:56<00:03, 608.53it/s]","\rparsing log, completed traces ::  94%|#########4| 29719/31509 [00:56<00:02, 619.87it/s]","\rparsing log, completed traces ::  95%|#########4| 29782/31509 [00:56<00:02, 604.27it/s]","\rparsing log, completed traces ::  95%|#########4| 29843/31509 [00:56<00:02, 602.57it/s]","\rparsing log, completed traces ::  95%|#########4| 29907/31509 [00:56<00:02, 612.61it/s]","\rparsing log, completed traces ::  95%|#########5| 29969/31509 [00:56<00:02, 612.57it/s]","\rparsing log, completed traces ::  95%|#########5| 30033/31509 [00:56<00:02, 618.02it/s]","\rparsing log, completed traces ::  96%|#########5| 30095/31509 [00:56<00:02, 617.30it/s]","\rparsing log, completed traces ::  96%|#########5| 30162/31509 [00:56<00:02, 628.57it/s]","\rparsing log, completed traces ::  96%|#########5| 30226/31509 [00:57<00:02, 629.89it/s]","\rparsing log, completed traces ::  96%|#########6| 30290/31509 [00:57<00:01, 626.91it/s]","\rparsing log, completed traces ::  96%|#########6| 30358/31509 [00:57<00:01, 641.17it/s]","\rparsing log, completed traces ::  97%|#########6| 30423/31509 [00:57<00:01, 641.53it/s]","\rparsing log, completed traces ::  97%|#########6| 30488/31509 [00:57<00:01, 630.38it/s]","\rparsing log, completed traces ::  97%|#########6| 30553/31509 [00:57<00:01, 634.80it/s]","\rparsing log, completed traces ::  97%|#########7| 30618/31509 [00:57<00:01, 639.13it/s]","\rparsing log, completed traces ::  97%|#########7| 30682/31509 [00:57<00:01, 629.35it/s]","\rparsing log, completed traces ::  98%|#########7| 30745/31509 [00:57<00:01, 627.15it/s]","\rparsing log, completed traces ::  98%|#########7| 30808/31509 [00:57<00:01, 627.84it/s]","\rparsing log, completed traces ::  98%|#########7| 30873/31509 [00:58<00:01, 634.08it/s]","\rparsing log, completed traces ::  98%|#########8| 30937/31509 [00:58<00:00, 628.06it/s]","\rparsing log, completed traces ::  98%|#########8| 31001/31509 [00:58<00:00, 630.55it/s]","\rparsing log, completed traces ::  99%|#########8| 31066/31509 [00:58<00:00, 634.85it/s]","\rparsing log, completed traces ::  99%|#########8| 31133/31509 [00:58<00:00, 642.05it/s]","\rparsing log, completed traces ::  99%|#########9| 31201/31509 [00:58<00:00, 652.68it/s]","\rparsing log, completed traces ::  99%|#########9| 31267/31509 [00:58<00:00, 620.64it/s]","\rparsing log, completed traces ::  99%|#########9| 31330/31509 [00:58<00:00, 604.81it/s]","\rparsing log, completed traces :: 100%|#########9| 31392/31509 [00:58<00:00, 608.34it/s]","\rparsing log, completed traces :: 100%|#########9| 31455/31509 [00:58<00:00, 613.41it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:59<00:00, 533.25it/s]","\n","[data] Loading XES: /workspace/data/Road_Traffic_Fine_Management_Process.xes","\n","\rparsing log, completed traces ::   0%|          | 0/150370 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 697/150370 [00:00<00:21, 6906.68it/s]","\rparsing log, completed traces ::   1%|          | 1465/150370 [00:00<00:20, 7359.53it/s]","\rparsing log, completed traces ::   1%|1         | 2202/150370 [00:00<00:38, 3859.15it/s]","\rparsing log, completed traces ::   2%|1         | 2933/150370 [00:00<00:31, 4731.19it/s]","\rparsing log, completed traces ::   2%|2         | 3646/150370 [00:00<00:27, 5369.82it/s]","\rparsing log, completed traces ::   3%|2         | 4402/150370 [00:00<00:24, 5959.62it/s]","\rparsing log, completed traces ::   3%|3         | 5147/150370 [00:00<00:22, 6361.55it/s]","\rparsing log, completed traces ::   4%|3         | 5890/150370 [00:01<00:21, 6667.40it/s]","\rparsing log, completed traces ::   4%|4         | 6616/150370 [00:01<00:21, 6838.86it/s]","\rparsing log, completed traces ::   5%|4         | 7347/150370 [00:01<00:20, 6976.70it/s]","\rparsing log, completed traces ::   5%|5         | 8110/150370 [00:01<00:19, 7168.14it/s]","\rparsing log, completed traces ::   6%|5         | 8898/150370 [00:01<00:19, 7357.41it/s]","\rparsing log, completed traces ::   6%|6         | 9675/150370 [00:01<00:18, 7456.07it/s]","\rparsing log, completed traces ::   7%|6         | 10429/150370 [00:01<00:18, 7468.83it/s]","\rparsing log, completed traces ::   7%|7         | 11182/150370 [00:01<00:31, 4395.65it/s]","\rparsing log, completed traces ::   8%|7         | 11927/150370 [00:02<00:27, 4995.45it/s]","\rparsing log, completed traces ::   8%|8         | 12656/150370 [00:02<00:25, 5503.84it/s]","\rparsing log, completed traces ::   9%|8         | 13400/150370 [00:02<00:22, 5969.46it/s]","\rparsing log, completed traces ::   9%|9         | 14125/150370 [00:02<00:21, 6295.04it/s]","\rparsing log, completed traces ::  10%|9         | 14863/150370 [00:02<00:20, 6565.89it/s]","\rparsing log, completed traces ::  10%|#         | 15628/150370 [00:02<00:19, 6864.89it/s]","\rparsing log, completed traces ::  11%|#         | 16409/150370 [00:02<00:18, 7109.29it/s]","\rparsing log, completed traces ::  11%|#1        | 17194/150370 [00:02<00:18, 7321.10it/s]","\rparsing log, completed traces ::  12%|#1        | 17964/150370 [00:02<00:17, 7422.62it/s]","\rparsing log, completed traces ::  12%|#2        | 18722/150370 [00:02<00:17, 7443.16it/s]","\rparsing log, completed traces ::  13%|#2        | 19477/150370 [00:03<00:17, 7451.09it/s]","\rparsing log, completed traces ::  13%|#3        | 20250/150370 [00:03<00:17, 7508.07it/s]","\rparsing log, completed traces ::  14%|#3        | 21006/150370 [00:03<00:31, 4154.18it/s]","\rparsing log, completed traces ::  15%|#4        | 21844/150370 [00:03<00:25, 4951.24it/s]","\rparsing log, completed traces ::  15%|#5        | 22672/150370 [00:03<00:22, 5658.65it/s]","\rparsing log, completed traces ::  16%|#5        | 23453/150370 [00:03<00:20, 6159.92it/s]","\rparsing log, completed traces ::  16%|#6        | 24187/150370 [00:03<00:20, 6271.53it/s]","\rparsing log, completed traces ::  17%|#6        | 24898/150370 [00:04<00:20, 6253.89it/s]","\rparsing log, completed traces ::  17%|#7        | 25582/150370 [00:04<00:19, 6293.28it/s]","\rparsing log, completed traces ::  17%|#7        | 26253/150370 [00:04<00:19, 6230.84it/s]","\rparsing log, completed traces ::  18%|#7        | 26905/150370 [00:04<00:20, 6152.04it/s]","\rparsing log, completed traces ::  18%|#8        | 27540/150370 [00:04<00:20, 6107.25it/s]","\rparsing log, completed traces ::  19%|#8        | 28165/150370 [00:04<00:19, 6133.93it/s]","\rparsing log, completed traces ::  19%|#9        | 28887/150370 [00:04<00:18, 6425.22it/s]","\rparsing log, completed traces ::  20%|#9        | 29633/150370 [00:04<00:17, 6722.42it/s]","\rparsing log, completed traces ::  20%|##        | 30313/150370 [00:04<00:18, 6615.24it/s]","\rparsing log, completed traces ::  21%|##        | 30980/150370 [00:04<00:18, 6514.20it/s]","\rparsing log, completed traces ::  21%|##1       | 31636/150370 [00:05<00:35, 3359.25it/s]","\rparsing log, completed traces ::  21%|##1       | 32264/150370 [00:05<00:30, 3871.53it/s]","\rparsing log, completed traces ::  22%|##1       | 32889/150370 [00:05<00:27, 4347.83it/s]","\rparsing log, completed traces ::  22%|##2       | 33520/150370 [00:05<00:24, 4783.88it/s]","\rparsing log, completed traces ::  23%|##2       | 34170/150370 [00:05<00:22, 5169.92it/s]","\rparsing log, completed traces ::  23%|##3       | 34784/150370 [00:05<00:21, 5417.30it/s]","\rparsing log, completed traces ::  24%|##3       | 35416/150370 [00:06<00:20, 5657.80it/s]","\rparsing log, completed traces ::  24%|##3       | 36054/150370 [00:06<00:19, 5856.37it/s]","\rparsing log, completed traces ::  24%|##4       | 36703/150370 [00:06<00:18, 6035.22it/s]","\rparsing log, completed traces ::  25%|##4       | 37332/150370 [00:06<00:18, 6107.06it/s]","\rparsing log, completed traces ::  25%|##5       | 37999/150370 [00:06<00:17, 6269.97it/s]","\rparsing log, completed traces ::  26%|##5       | 38672/150370 [00:06<00:17, 6405.12it/s]","\rparsing log, completed traces ::  26%|##6       | 39350/150370 [00:06<00:17, 6515.41it/s]","\rparsing log, completed traces ::  27%|##6       | 40010/150370 [00:06<00:16, 6540.36it/s]","\rparsing log, completed traces ::  27%|##7       | 40675/150370 [00:06<00:16, 6562.43it/s]","\rparsing log, completed traces ::  28%|##7       | 41376/150370 [00:06<00:16, 6695.07it/s]","\rparsing log, completed traces ::  28%|##7       | 42076/150370 [00:07<00:15, 6784.87it/s]","\rparsing log, completed traces ::  28%|##8       | 42757/150370 [00:07<00:34, 3092.05it/s]","\rparsing log, completed traces ::  29%|##8       | 43450/150370 [00:07<00:28, 3716.74it/s]","\rparsing log, completed traces ::  29%|##9       | 44138/150370 [00:07<00:24, 4306.10it/s]","\rparsing log, completed traces ::  30%|##9       | 44831/150370 [00:07<00:21, 4861.55it/s]","\rparsing log, completed traces ::  30%|###       | 45505/150370 [00:07<00:19, 5298.08it/s]","\rparsing log, completed traces ::  31%|###       | 46167/150370 [00:08<00:18, 5626.10it/s]","\rparsing log, completed traces ::  31%|###1      | 46833/150370 [00:08<00:17, 5896.89it/s]","\rparsing log, completed traces ::  32%|###1      | 47487/150370 [00:08<00:17, 6050.16it/s]","\rparsing log, completed traces ::  32%|###2      | 48185/150370 [00:08<00:16, 6302.30it/s]","\rparsing log, completed traces ::  33%|###2      | 48966/150370 [00:08<00:15, 6730.51it/s]","\rparsing log, completed traces ::  33%|###3      | 49669/150370 [00:08<00:14, 6814.99it/s]","\rparsing log, completed traces ::  33%|###3      | 50370/150370 [00:08<00:14, 6815.79it/s]","\rparsing log, completed traces ::  34%|###3      | 51065/150370 [00:08<00:14, 6774.87it/s]","\rparsing log, completed traces ::  34%|###4      | 51755/150370 [00:08<00:14, 6810.88it/s]","\rparsing log, completed traces ::  35%|###4      | 52443/150370 [00:08<00:14, 6743.02it/s]","\rparsing log, completed traces ::  35%|###5      | 53127/150370 [00:09<00:14, 6754.36it/s]","\rparsing log, completed traces ::  36%|###5      | 53806/150370 [00:09<00:14, 6735.47it/s]","\rparsing log, completed traces ::  36%|###6      | 54567/150370 [00:09<00:13, 6985.86it/s]","\rparsing log, completed traces ::  37%|###6      | 55268/150370 [00:09<00:13, 6986.56it/s]","\rparsing log, completed traces ::  37%|###7      | 55970/150370 [00:09<00:13, 6995.91it/s]","\rparsing log, completed traces ::  38%|###7      | 56671/150370 [00:09<00:29, 3132.10it/s]","\rparsing log, completed traces ::  38%|###8      | 57382/150370 [00:10<00:24, 3768.82it/s]","\rparsing log, completed traces ::  39%|###8      | 58076/150370 [00:10<00:21, 4358.76it/s]","\rparsing log, completed traces ::  39%|###9      | 58778/150370 [00:10<00:18, 4908.79it/s]","\rparsing log, completed traces ::  40%|###9      | 59496/150370 [00:10<00:16, 5418.21it/s]","\rparsing log, completed traces ::  40%|####      | 60221/150370 [00:10<00:15, 5856.17it/s]","\rparsing log, completed traces ::  41%|####      | 60991/150370 [00:10<00:14, 6333.74it/s]","\rparsing log, completed traces ::  41%|####1     | 61798/150370 [00:10<00:13, 6785.35it/s]","\rparsing log, completed traces ::  42%|####1     | 62554/150370 [00:10<00:12, 6999.50it/s]","\rparsing log, completed traces ::  42%|####2     | 63295/150370 [00:10<00:12, 7027.60it/s]","\rparsing log, completed traces ::  43%|####2     | 64026/150370 [00:10<00:12, 7068.51it/s]","\rparsing log, completed traces ::  43%|####3     | 64753/150370 [00:11<00:12, 7115.96it/s]","\rparsing log, completed traces ::  44%|####3     | 65479/150370 [00:11<00:11, 7152.77it/s]","\rparsing log, completed traces ::  44%|####4     | 66237/150370 [00:11<00:11, 7265.11it/s]","\rparsing log, completed traces ::  45%|####4     | 66981/150370 [00:11<00:11, 7316.25it/s]","\rparsing log, completed traces ::  45%|####5     | 67815/150370 [00:11<00:10, 7619.69it/s]","\rparsing log, completed traces ::  46%|####5     | 68581/150370 [00:11<00:10, 7525.90it/s]","\rparsing log, completed traces ::  46%|####6     | 69337/150370 [00:11<00:10, 7509.47it/s]","\rparsing log, completed traces ::  47%|####6     | 70090/150370 [00:11<00:10, 7399.02it/s]","\rparsing log, completed traces ::  47%|####7     | 70832/150370 [00:11<00:10, 7352.11it/s]","\rparsing log, completed traces ::  48%|####7     | 71569/150370 [00:11<00:10, 7253.31it/s]","\rparsing log, completed traces ::  48%|####8     | 72296/150370 [00:12<00:10, 7214.56it/s]","\rparsing log, completed traces ::  49%|####8     | 73019/150370 [00:12<00:10, 7090.06it/s]","\rparsing log, completed traces ::  49%|####9     | 73729/150370 [00:12<00:11, 6956.60it/s]","\rparsing log, completed traces ::  49%|####9     | 74426/150370 [00:12<00:26, 2907.02it/s]","\rparsing log, completed traces ::  50%|#####     | 75214/150370 [00:12<00:20, 3637.72it/s]","\rparsing log, completed traces ::  51%|#####     | 75953/150370 [00:13<00:17, 4289.97it/s]","\rparsing log, completed traces ::  51%|#####1    | 76728/150370 [00:13<00:14, 4978.58it/s]","\rparsing log, completed traces ::  52%|#####1    | 77441/150370 [00:13<00:13, 5454.82it/s]","\rparsing log, completed traces ::  52%|#####1    | 78143/150370 [00:13<00:12, 5819.86it/s]","\rparsing log, completed traces ::  52%|#####2    | 78857/150370 [00:13<00:11, 6156.22it/s]","\rparsing log, completed traces ::  53%|#####2    | 79564/150370 [00:13<00:11, 6400.03it/s]","\rparsing log, completed traces ::  53%|#####3    | 80282/150370 [00:13<00:10, 6614.51it/s]","\rparsing log, completed traces ::  54%|#####3    | 80989/150370 [00:13<00:10, 6688.96it/s]","\rparsing log, completed traces ::  54%|#####4    | 81690/150370 [00:13<00:10, 6758.93it/s]","\rparsing log, completed traces ::  55%|#####4    | 82473/150370 [00:13<00:09, 7067.78it/s]","\rparsing log, completed traces ::  55%|#####5    | 83232/150370 [00:14<00:09, 7220.31it/s]","\rparsing log, completed traces ::  56%|#####5    | 83967/150370 [00:14<00:09, 7238.32it/s]","\rparsing log, completed traces ::  56%|#####6    | 84700/150370 [00:14<00:09, 7228.92it/s]","\rparsing log, completed traces ::  57%|#####6    | 85429/150370 [00:14<00:09, 7209.62it/s]","\rparsing log, completed traces ::  57%|#####7    | 86162/150370 [00:14<00:08, 7243.36it/s]","\rparsing log, completed traces ::  58%|#####7    | 86890/150370 [00:14<00:08, 7244.45it/s]","\rparsing log, completed traces ::  58%|#####8    | 87630/150370 [00:14<00:08, 7267.64it/s]","\rparsing log, completed traces ::  59%|#####8    | 88385/150370 [00:14<00:08, 7340.07it/s]","\rparsing log, completed traces ::  59%|#####9    | 89121/150370 [00:14<00:08, 7339.60it/s]","\rparsing log, completed traces ::  60%|#####9    | 89883/150370 [00:14<00:08, 7422.00it/s]","\rparsing log, completed traces ::  60%|######    | 90626/150370 [00:15<00:08, 7408.03it/s]","\rparsing log, completed traces ::  61%|######    | 91374/150370 [00:15<00:07, 7429.22it/s]","\rparsing log, completed traces ::  61%|######1   | 92118/150370 [00:15<00:07, 7417.33it/s]","\rparsing log, completed traces ::  62%|######1   | 92860/150370 [00:15<00:07, 7372.25it/s]","\rparsing log, completed traces ::  62%|######2   | 93598/150370 [00:15<00:07, 7368.78it/s]","\rparsing log, completed traces ::  63%|######2   | 94336/150370 [00:16<00:21, 2667.38it/s]","\rparsing log, completed traces ::  63%|######3   | 95095/150370 [00:16<00:16, 3324.78it/s]","\rparsing log, completed traces ::  64%|######3   | 95844/150370 [00:16<00:13, 3991.68it/s]","\rparsing log, completed traces ::  64%|######4   | 96561/150370 [00:16<00:11, 4583.59it/s]","\rparsing log, completed traces ::  65%|######4   | 97291/150370 [00:16<00:10, 5148.07it/s]","\rparsing log, completed traces ::  65%|######5   | 98026/150370 [00:16<00:09, 5655.61it/s]","\rparsing log, completed traces ::  66%|######5   | 98761/150370 [00:16<00:08, 6075.23it/s]","\rparsing log, completed traces ::  66%|######6   | 99497/150370 [00:16<00:07, 6409.22it/s]","\rparsing log, completed traces ::  67%|######6   | 100216/150370 [00:16<00:07, 6592.01it/s]","\rparsing log, completed traces ::  67%|######7   | 100968/150370 [00:17<00:07, 6850.53it/s]","\rparsing log, completed traces ::  68%|######7   | 101704/150370 [00:17<00:06, 6993.76it/s]","\rparsing log, completed traces ::  68%|######8   | 102439/150370 [00:17<00:06, 7095.97it/s]","\rparsing log, completed traces ::  69%|######8   | 103170/150370 [00:17<00:06, 7158.20it/s]","\rparsing log, completed traces ::  69%|######9   | 103901/150370 [00:17<00:06, 7167.31it/s]","\rparsing log, completed traces ::  70%|######9   | 104629/150370 [00:17<00:06, 7158.19it/s]","\rparsing log, completed traces ::  70%|#######   | 105353/150370 [00:17<00:06, 7157.37it/s]","\rparsing log, completed traces ::  71%|#######   | 106074/150370 [00:17<00:06, 7126.54it/s]","\rparsing log, completed traces ::  71%|#######1  | 106793/150370 [00:17<00:06, 7144.19it/s]","\rparsing log, completed traces ::  71%|#######1  | 107510/150370 [00:18<00:05, 7147.74it/s]","\rparsing log, completed traces ::  72%|#######1  | 108227/150370 [00:18<00:05, 7079.89it/s]","\rparsing log, completed traces ::  72%|#######2  | 108942/150370 [00:18<00:05, 7100.07it/s]","\rparsing log, completed traces ::  73%|#######2  | 109653/150370 [00:18<00:05, 7095.63it/s]","\rparsing log, completed traces ::  73%|#######3  | 110364/150370 [00:18<00:05, 7081.78it/s]","\rparsing log, completed traces ::  74%|#######3  | 111081/150370 [00:18<00:05, 7107.08it/s]","\rparsing log, completed traces ::  74%|#######4  | 111793/150370 [00:18<00:05, 7108.89it/s]","\rparsing log, completed traces ::  75%|#######4  | 112505/150370 [00:18<00:05, 7083.76it/s]","\rparsing log, completed traces ::  75%|#######5  | 113214/150370 [00:18<00:05, 6981.04it/s]","\rparsing log, completed traces ::  76%|#######5  | 114044/150370 [00:18<00:04, 7370.33it/s]","\rparsing log, completed traces ::  76%|#######6  | 114919/150370 [00:19<00:04, 7762.50it/s]","\rparsing log, completed traces ::  77%|#######7  | 115803/150370 [00:19<00:04, 8081.81it/s]","\rparsing log, completed traces ::  78%|#######7  | 116701/150370 [00:19<00:04, 8348.75it/s]","\rparsing log, completed traces ::  78%|#######8  | 117575/150370 [00:19<00:11, 2874.12it/s]","\rparsing log, completed traces ::  79%|#######8  | 118235/150370 [00:20<00:09, 3345.50it/s]","\rparsing log, completed traces ::  79%|#######9  | 118899/150370 [00:20<00:08, 3844.15it/s]","\rparsing log, completed traces ::  80%|#######9  | 119560/150370 [00:20<00:07, 4339.23it/s]","\rparsing log, completed traces ::  80%|#######9  | 120210/150370 [00:20<00:06, 4776.35it/s]","\rparsing log, completed traces ::  80%|########  | 120864/150370 [00:20<00:05, 5171.79it/s]","\rparsing log, completed traces ::  81%|########  | 121526/150370 [00:20<00:05, 5524.06it/s]","\rparsing log, completed traces ::  81%|########1 | 122197/150370 [00:20<00:04, 5829.87it/s]","\rparsing log, completed traces ::  82%|########1 | 122878/150370 [00:20<00:04, 6085.09it/s]","\rparsing log, completed traces ::  82%|########2 | 123542/150370 [00:20<00:04, 6235.26it/s]","\rparsing log, completed traces ::  83%|########2 | 124206/150370 [00:20<00:04, 6314.07it/s]","\rparsing log, completed traces ::  83%|########3 | 124866/150370 [00:21<00:03, 6379.54it/s]","\rparsing log, completed traces ::  83%|########3 | 125537/150370 [00:21<00:03, 6474.85it/s]","\rparsing log, completed traces ::  84%|########3 | 126217/150370 [00:21<00:03, 6564.09it/s]","\rparsing log, completed traces ::  84%|########4 | 126919/150370 [00:21<00:03, 6698.47it/s]","\rparsing log, completed traces ::  85%|########4 | 127597/150370 [00:21<00:03, 6704.71it/s]","\rparsing log, completed traces ::  85%|########5 | 128276/150370 [00:21<00:03, 6729.95it/s]","\rparsing log, completed traces ::  86%|########5 | 128961/150370 [00:21<00:03, 6754.82it/s]","\rparsing log, completed traces ::  86%|########6 | 129655/150370 [00:21<00:03, 6806.68it/s]","\rparsing log, completed traces ::  87%|########6 | 130341/150370 [00:21<00:02, 6811.27it/s]","\rparsing log, completed traces ::  87%|########7 | 131037/150370 [00:21<00:02, 6855.19it/s]","\rparsing log, completed traces ::  88%|########7 | 131767/150370 [00:22<00:02, 6986.18it/s]","\rparsing log, completed traces ::  88%|########8 | 132578/150370 [00:22<00:02, 7321.84it/s]","\rparsing log, completed traces ::  89%|########8 | 133344/150370 [00:22<00:02, 7421.64it/s]","\rparsing log, completed traces ::  89%|########9 | 134087/150370 [00:22<00:02, 7319.69it/s]","\rparsing log, completed traces ::  90%|########9 | 134820/150370 [00:22<00:02, 7229.42it/s]","\rparsing log, completed traces ::  90%|######### | 135544/150370 [00:22<00:02, 7156.12it/s]","\rparsing log, completed traces ::  91%|######### | 136261/150370 [00:22<00:01, 7064.68it/s]","\rparsing log, completed traces ::  91%|#########1| 136968/150370 [00:22<00:01, 7049.79it/s]","\rparsing log, completed traces ::  92%|#########1| 137674/150370 [00:22<00:01, 6934.01it/s]","\rparsing log, completed traces ::  92%|#########2| 138379/150370 [00:22<00:01, 6966.99it/s]","\rparsing log, completed traces ::  92%|#########2| 139090/150370 [00:23<00:01, 7008.96it/s]","\rparsing log, completed traces ::  93%|#########2| 139792/150370 [00:23<00:01, 6958.58it/s]","\rparsing log, completed traces ::  93%|#########3| 140520/150370 [00:23<00:01, 7052.21it/s]","\rparsing log, completed traces ::  94%|#########3| 141226/150370 [00:23<00:01, 7043.16it/s]","\rparsing log, completed traces ::  94%|#########4| 141954/150370 [00:23<00:01, 7113.23it/s]","\rparsing log, completed traces ::  95%|#########4| 142666/150370 [00:23<00:01, 7092.79it/s]","\rparsing log, completed traces ::  95%|#########5| 143376/150370 [00:24<00:03, 2221.51it/s]","\rparsing log, completed traces ::  96%|#########5| 144094/150370 [00:24<00:02, 2802.94it/s]","\rparsing log, completed traces ::  96%|#########6| 144744/150370 [00:24<00:01, 3334.15it/s]","\rparsing log, completed traces ::  97%|#########6| 145345/150370 [00:24<00:01, 3782.98it/s]","\rparsing log, completed traces ::  97%|#########7| 145944/150370 [00:24<00:01, 4156.64it/s]","\rparsing log, completed traces ::  97%|#########7| 146531/150370 [00:24<00:00, 4452.47it/s]","\rparsing log, completed traces ::  98%|#########7| 147106/150370 [00:25<00:00, 4730.26it/s]","\rparsing log, completed traces ::  98%|#########8| 147677/150370 [00:25<00:00, 4938.23it/s]","\rparsing log, completed traces ::  99%|#########8| 148244/150370 [00:25<00:00, 5120.96it/s]","\rparsing log, completed traces ::  99%|#########8| 148840/150370 [00:25<00:00, 5347.30it/s]","\rparsing log, completed traces ::  99%|#########9| 149445/150370 [00:25<00:00, 5535.73it/s]","\rparsing log, completed traces :: 100%|#########9| 150103/150370 [00:25<00:00, 5826.03it/s]","","\rparsing log, completed traces :: 100%|##########| 150370/150370 [00:25<00:00, 5872.98it/s]","\n","[data] Loaded datasets: ['BPI2012', 'BPI2017', 'ROAD']","\n","[data] Using dataset: BPI2017, shape=(1202267, 5)","\n","[split] Cases: train=22056, val=4726, test=4727","\n","[samples] train=311338, val=65786, test=66673","\n","Epoch 1: train_loss = 0.7316 | train_acc = 0.8066 | train_f1 = 0.5894 | train_top3 = 0.9767","\n","Epoch 1: validation_loss = 0.5268 | val_acc = 0.7886 | val_f1 = 0.5650 | val_top3 = 0.9732","\n","Epoch 2: train_loss = 0.4809 | train_acc = 0.8123 | train_f1 = 0.6096 | train_top3 = 0.9787","\n","Epoch 2: validation_loss = 0.5085 | val_acc = 0.7919 | val_f1 = 0.5859 | val_top3 = 0.9783","\n","Epoch 3: train_loss = 0.4672 | train_acc = 0.8064 | train_f1 = 0.6181 | train_top3 = 0.9798","\n","Epoch 3: validation_loss = 0.5003 | val_acc = 0.7905 | val_f1 = 0.6021 | val_top3 = 0.9797","\n","Epoch 4: train_loss = 0.4597 | train_acc = 0.8125 | train_f1 = 0.6138 | train_top3 = 0.9798","\n","Epoch 4: validation_loss = 0.4928 | val_acc = 0.7945 | val_f1 = 0.5967 | val_top3 = 0.9774","\n","Epoch 5: train_loss = 0.4532 | train_acc = 0.8173 | train_f1 = 0.6446 | train_top3 = 0.9811","\n","Epoch 5: validation_loss = 0.4914 | val_acc = 0.7959 | val_f1 = 0.6286 | val_top3 = 0.9815","\n","Epoch 6: train_loss = 0.4500 | train_acc = 0.8179 | train_f1 = 0.6423 | train_top3 = 0.9809","\n","Epoch 6: validation_loss = 0.4961 | val_acc = 0.7969 | val_f1 = 0.6242 | val_top3 = 0.9808","\n","Epoch 7: train_loss = 0.4468 | train_acc = 0.8173 | train_f1 = 0.6490 | train_top3 = 0.9817","\n","Epoch 7: validation_loss = 0.4924 | val_acc = 0.7959 | val_f1 = 0.6310 | val_top3 = 0.9791","\n","Epoch 8: train_loss = 0.4438 | train_acc = 0.8206 | train_f1 = 0.6641 | train_top3 = 0.9824","\n","Epoch 8: validation_loss = 0.4814 | val_acc = 0.8003 | val_f1 = 0.6334 | val_top3 = 0.9805","\n","[final] Test: loss=0.4961 | acc=0.7952 | macro_f1=0.6316 | top3_acc=0.9776","\n","Top-3 Next-Activity Accuracy (test) = 0.9776","\n","Execution time: 5 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"The code will load the experiment_data.npy from the working directory and iterate over each dataset contained in the saved dictionary. For training and validation, it will compute and print the best values across epochs for loss (minimum), accuracy (maximum), macro F1 score (maximum), and top-3 accuracy (maximum). For the test split, it will print the final metrics that were recorded. The script follows the required structure, avoids plots, and prints clearly labeled metrics for each dataset.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef load_experiment_data(working_dir):\n    path = os.path.join(working_dir, \"experiment_data.npy\")\n    if not os.path.exists(path):\n        raise FileNotFoundError(f\"Could not find experiment_data.npy in {working_dir}\")\n    data = np.load(path, allow_pickle=True).item()\n    if not isinstance(data, dict):\n        raise ValueError(\"Loaded experiment data is not a dictionary\")\n    return data\n\n\ndef best_by_min(pairs):\n    # pairs: list of (epoch_or_tag, value)\n    if not pairs:\n        return None\n    # filter numeric epochs\n    numeric = [\n        (e, v)\n        for (e, v) in pairs\n        if isinstance(e, (int, float, np.integer, np.floating))\n    ]\n    if not numeric:\n        # if only tags like 'final', take min over all by value anyway\n        return min(pairs, key=lambda x: x[1])\n    return min(numeric, key=lambda x: x[1])\n\n\ndef best_by_max_from_metric_tuples(tuples, idx):\n    # tuples: list of (epoch, acc, f1, top3); pick max over metric at given idx\n    if not tuples:\n        return None\n    numeric = [\n        t for t in tuples if isinstance(t[0], (int, float, np.integer, np.floating))\n    ]\n    source = numeric if numeric else tuples\n    # return (epoch_or_tag, value)\n    # Keep the epoch/tag along with the value if needed; here we return the value only\n    best_tuple = max(source, key=lambda t: t[idx])\n    return best_tuple[0], best_tuple[idx]\n\n\ndef extract_and_print_metrics(experiment_data):\n    # Iterate over datasets\n    for dataset_key, content in experiment_data.items():\n        # Determine dataset name to print\n        dataset_name = content.get(\"meta\", {}).get(\"dataset_name\", dataset_key)\n        print(f\"Dataset: {dataset_name}\")\n\n        # Losses\n        losses = content.get(\"losses\", {})\n        train_losses = losses.get(\"train\", [])  # list of (epoch, loss)\n        val_losses = losses.get(\"val\", [])  # list of (epoch, loss)\n\n        best_train_loss = best_by_min(train_losses)\n        best_val_loss = best_by_min(val_losses)\n\n        if best_train_loss is not None:\n            print(f\"train loss: {best_train_loss[1]:.6f}\")\n        if best_val_loss is not None:\n            print(f\"validation loss: {best_val_loss[1]:.6f}\")\n\n        # Metrics\n        metrics = content.get(\"metrics\", {})\n        train_metrics = metrics.get(\"train\", [])  # list of (epoch, acc, f1, top3)\n        val_metrics = metrics.get(\"val\", [])  # list of (epoch, acc, f1, top3)\n        test_metrics = metrics.get(\"test\", [])  # typically [(\"final\", acc, f1, top3)]\n\n        # Best train metrics\n        if train_metrics:\n            # indices: 1=acc, 2=f1, 3=top3\n            _, best_tr_acc = best_by_max_from_metric_tuples(train_metrics, 1)\n            _, best_tr_f1 = best_by_max_from_metric_tuples(train_metrics, 2)\n            _, best_tr_top3 = best_by_max_from_metric_tuples(train_metrics, 3)\n            print(f\"train accuracy: {best_tr_acc:.6f}\")\n            print(f\"train macro F1 score: {best_tr_f1:.6f}\")\n            print(f\"train top-3 accuracy: {best_tr_top3:.6f}\")\n\n        # Best validation metrics\n        if val_metrics:\n            _, best_val_acc = best_by_max_from_metric_tuples(val_metrics, 1)\n            _, best_val_f1 = best_by_max_from_metric_tuples(val_metrics, 2)\n            _, best_val_top3 = best_by_max_from_metric_tuples(val_metrics, 3)\n            print(f\"validation accuracy: {best_val_acc:.6f}\")\n            print(f\"validation macro F1 score: {best_val_f1:.6f}\")\n            print(f\"validation top-3 accuracy: {best_val_top3:.6f}\")\n\n        # Final test metrics\n        # Prefer the entry tagged 'final' if present; otherwise, use the last one\n        test_acc = test_f1 = test_top3 = None\n        if test_metrics:\n            final_entry = None\n            for entry in test_metrics:\n                if isinstance(entry[0], str) and entry[0].lower() == \"final\":\n                    final_entry = entry\n                    break\n            if final_entry is None:\n                final_entry = test_metrics[-1]\n            # entry: (\"final\" or epoch, acc, f1, top3)\n            if len(final_entry) >= 4:\n                test_acc = final_entry[1]\n                test_f1 = final_entry[2]\n                test_top3 = final_entry[3]\n        # Fallback to meta if available\n        meta = content.get(\"meta\", {})\n        if test_acc is None and \"test_accuracy\" in meta:\n            test_acc = meta[\"test_accuracy\"]\n        if test_f1 is None and \"test_macro_f1\" in meta:\n            test_f1 = meta[\"test_macro_f1\"]\n        if test_top3 is None and \"test_top3\" in meta:\n            test_top3 = meta[\"test_top3\"]\n\n        if test_acc is not None:\n            print(f\"test accuracy: {float(test_acc):.6f}\")\n        if test_f1 is not None:\n            print(f\"test macro F1 score: {float(test_f1):.6f}\")\n        if test_top3 is not None:\n            print(f\"test top-3 accuracy: {float(test_top3):.6f}\")\n\n\ndef main():\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    experiment_data = load_experiment_data(working_dir)\n    extract_and_print_metrics(experiment_data)\n\n\n# Execute immediately (no __main__ guard as per instructions)\nmain()\n","parse_term_out":["Dataset: BPI2017","\n","train loss: 0.443762","\n","validation loss: 0.481377","\n","train accuracy: 0.820633","\n","train macro F1 score: 0.664127","\n","train top-3 accuracy: 0.982383","\n","validation accuracy: 0.800267","\n","validation macro F1 score: 0.633376","\n","validation top-3 accuracy: 0.981533","\n","test accuracy: 0.795200","\n","test macro F1 score: 0.631615","\n","test top-3 accuracy: 0.977600","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":346.8022162914276,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"","exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_3645f80923974ef6a45cb404d8ddc8bd_proc_332089","metric":{"value":{"metric_names":[{"metric_name":"train loss","lower_is_better":true,"description":"Cross-entropy loss on the training split; lower values indicate better fit.","data":[{"dataset_name":"BPI2017","final_value":0.443762,"best_value":0.443762}]},{"metric_name":"validation loss","lower_is_better":true,"description":"Cross-entropy loss on the validation split; used for model selection.","data":[{"dataset_name":"BPI2017","final_value":0.481377,"best_value":0.481377}]},{"metric_name":"train accuracy","lower_is_better":false,"description":"Classification accuracy on the training split.","data":[{"dataset_name":"BPI2017","final_value":0.820633,"best_value":0.820633}]},{"metric_name":"train macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the training split across all classes.","data":[{"dataset_name":"BPI2017","final_value":0.664127,"best_value":0.664127}]},{"metric_name":"train top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the training split (correct if true label in top 3 predictions).","data":[{"dataset_name":"BPI2017","final_value":0.982383,"best_value":0.982383}]},{"metric_name":"validation accuracy","lower_is_better":false,"description":"Classification accuracy on the validation split.","data":[{"dataset_name":"BPI2017","final_value":0.800267,"best_value":0.800267}]},{"metric_name":"validation macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the validation split across all classes.","data":[{"dataset_name":"BPI2017","final_value":0.633376,"best_value":0.633376}]},{"metric_name":"validation top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the validation split (correct if true label in top 3 predictions).","data":[{"dataset_name":"BPI2017","final_value":0.981533,"best_value":0.981533}]},{"metric_name":"test accuracy","lower_is_better":false,"description":"Classification accuracy on the held-out test split.","data":[{"dataset_name":"BPI2017","final_value":0.7952,"best_value":0.7952}]},{"metric_name":"test macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the held-out test split across all classes.","data":[{"dataset_name":"BPI2017","final_value":0.631615,"best_value":0.631615}]},{"metric_name":"test top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the held-out test split (correct if true label in top 3 predictions).","data":[{"dataset_name":"BPI2017","final_value":0.9776,"best_value":0.9776}]}]},"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":null,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":[],"plot_paths":[],"plot_analyses":[],"vlm_feedback_summary":[],"datasets_successfully_tested":[],"ablation_name":null,"hyperparam_name":null,"is_seed_node":false,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"import os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport warnings\n\nwarnings.filterwarnings(\"ignore\")\nimport numpy as np\nimport pandas as pd\nimport matplotlib.pyplot as plt\nfrom pathlib import Path\nfrom datetime import timedelta\nfrom collections import defaultdict\nfrom typing import List, Dict, Tuple\nimport torch\nimport torch.nn as nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nfrom sklearn.preprocessing import StandardScaler\n\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data container\nexperiment_data = {\n    \"DEFAULT\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n        \"top3_history\": {\"train\": [], \"val\": []},\n    }\n}\n\n\n# ------------------ Data discovery and loading ------------------ #\ndef _has_xes(dirpath: Path) -> bool:\n    try:\n        return dirpath.is_dir() and (\n            any(dirpath.glob(\"*.xes\")) or any(dirpath.glob(\"*.xes.gz\"))\n        )\n    except Exception:\n        return False\n\n\ndef _resolve_data_dir() -> Path:\n    candidates: List[Path] = []\n    candidates += [Path(\"input\").resolve(), (Path.cwd() / \"input\").resolve()]\n    cwd = Path.cwd().resolve()\n    for base in [cwd, *cwd.parents]:\n        candidates.append((base / \"data\").resolve())\n        candidates.append((base / \"input\").resolve())\n    candidates += [\n        Path(\"/workspace/input\"),\n        Path(\"/workspace/data\"),\n        Path(\"/workspace/ai_scientist/data\"),\n        Path(\"/workspace/AI-Scientist-v2/data\"),\n        Path(\"/workspace/experiments/data\"),\n        Path(\"/workspace/ai_scientist/input\"),\n        Path(\"/workspace/experiments/input\"),\n    ]\n    seen = set()\n    for p in candidates:\n        if p in seen:\n            continue\n        seen.add(p)\n        if _has_xes(p):\n            print(f\"[data] Using discovered data dir: {p}\")\n            return p\n    raise FileNotFoundError(\"No .xes found\")\n\n\ndef xes_to_df(xes_path: Path) -> pd.DataFrame:\n    try:\n        from pm4py.objects.log.importer.xes import importer as xes_importer\n    except Exception as e:\n        raise ImportError(\"pm4py is required. pip install pm4py\") from e\n    print(f\"[data] Loading XES: {xes_path}\")\n    log = xes_importer.apply(str(xes_path))\n    rows = []\n    for tr in log:\n        case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n            \"case:concept:name\"\n        )\n        for e in tr:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": e.get(\"concept:name\"),\n                    \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                    \"timestamp\": e.get(\"time:timestamp\"),\n                    \"resource\": e.get(\"org:resource\", \"System\"),\n                }\n            )\n    df = pd.DataFrame(rows)\n    df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n    df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n    df = df.sort_values([\"timestamp\", \"case_id\"]).reset_index(drop=True)\n    return df\n\n\ndef load_default_dataset() -> Tuple[str, pd.DataFrame]:\n    datasets = {}\n    try:\n        d = _resolve_data_dir()\n        files = list(d.glob(\"*.xes\")) + list(d.glob(\"*.xes.gz\"))\n        # Prefer BPI 2017, then BPI 2012, then Road\n        pref = [\n            \"BPI_Challenge_2017\",\n            \"BPI2017\",\n            \"2017\",\n            \"BPI_Challenge_2012\",\n            \"BPI2012\",\n            \"2012\",\n            \"Road_Traffic_Fine_Management_Process\",\n            \"Traffic\",\n            \"Fine\",\n            \"Road\",\n        ]\n        chosen = None\n        for p in pref:\n            for f in files:\n                if p in f.name:\n                    chosen = f\n                    break\n            if chosen is not None:\n                break\n        if chosen is None and files:\n            chosen = files[0]\n        if chosen is not None:\n            df = xes_to_df(chosen)\n            name = (\n                \"BPI2017\"\n                if \"2017\" in chosen.name\n                else (\"BPI2012\" if \"2012\" in chosen.name else \"ROAD\")\n            )\n            return name, df\n    except Exception as e:\n        print(f\"[warn] Data discovery failed: {e}\")\n    # Fallback: synthetic dataset\n    print(\"[data] Generating synthetic log\")\n    rng = np.random.RandomState(42)\n    activities = [\"A\", \"B\", \"C\", \"D\", \"E\", \"F\"]\n    resources = [\"R1\", \"R2\", \"R3\"]\n    rows = []\n    start = pd.Timestamp(\"2020-01-01\", tz=\"UTC\")\n    n_cases = 200\n    for c in range(n_cases):\n        case_id = f\"C{c:04d}\"\n        t = start + pd.Timedelta(days=int(c / 5))\n        length = rng.randint(4, 9)\n        # Simple branching: A -> (B or C) -> D/E -> F\n        seq = [\"A\"]\n        for i in range(length - 1):\n            if seq[-1] == \"A\":\n                nxt = \"B\" if rng.rand() < 0.6 else \"C\"\n            elif seq[-1] in [\"B\", \"C\"]:\n                nxt = \"D\" if rng.rand() < 0.5 else \"E\"\n            elif seq[-1] in [\"D\", \"E\"]:\n                nxt = \"F\" if rng.rand() < 0.7 else rng.choice([\"B\", \"C\"])\n            else:\n                nxt = rng.choice(activities)\n            seq.append(nxt)\n        for act in seq:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": act,\n                    \"lifecycle\": \"complete\",\n                    \"timestamp\": t,\n                    \"resource\": rng.choice(resources),\n                }\n            )\n            t += pd.Timedelta(minutes=int(rng.exponential(120)))\n    df = pd.DataFrame(rows).sort_values([\"timestamp\", \"case_id\"]).reset_index(drop=True)\n    return \"SYNTH\", df\n\n\n# ------------------ Prefix dataset building ------------------ #\ndef build_prefix_samples(\n    df: pd.DataFrame, max_len: int = 10\n) -> Tuple[pd.DataFrame, Dict[str, int]]:\n    # Filter lifecycle=complete if present\n    if \"lifecycle\" in df.columns:\n        # keep all for simplicity\n        pass\n    # build traces\n    g = df.sort_values(\"timestamp\").groupby(\"case_id\")\n    traces = []\n    for cid, grp in g:\n        acts = grp[\"activity\"].tolist()\n        times = grp[\"timestamp\"].tolist()\n        res = (\n            grp[\"resource\"].tolist()\n            if \"resource\" in grp.columns\n            else [\"System\"] * len(acts)\n        )\n        traces.append((cid, acts, times, res))\n    # vocab\n    act_set = sorted({a for _, acts, _, _ in traces for a in acts})\n    act2ix = {a: i + 1 for i, a in enumerate(act_set)}  # 0 is PAD\n    samples = []\n    for cid, acts, times, res in traces:\n        if len(acts) < 2:\n            continue\n        for i in range(\n            1, min(len(acts), max_len + 1) - 0\n        ):  # consider prefixes up to max_len\n            if i >= len(acts):\n                break\n            prefix_acts = acts[:i]\n            next_act = acts[i] if i < len(acts) else None\n            if next_act is None:\n                continue\n            prefix_times = times[:i]\n            # temporal features from last event in prefix\n            start_t = prefix_times[0]\n            last_t = prefix_times[-1]\n            delta_start = (last_t - start_t).total_seconds() / 3600.0\n            if len(prefix_times) >= 2:\n                delta_last = (\n                    prefix_times[-1] - prefix_times[-2]\n                ).total_seconds() / 3600.0\n            else:\n                delta_last = 0.0\n            hour = last_t.hour\n            weekday = last_t.weekday()\n            working = 1.0 if (weekday < 5 and 8 <= hour < 18) else 0.0\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq\": [act2ix[a] for a in prefix_acts][-max_len:],\n                    \"len\": min(len(prefix_acts), max_len),\n                    \"aux\": np.array(\n                        [delta_start, delta_last, hour, weekday, working],\n                        dtype=np.float32,\n                    ),\n                    \"y\": act2ix[next_act],\n                }\n            )\n    data = pd.DataFrame(samples)\n    return data, act2ix\n\n\ndef time_based_split(\n    df: pd.DataFrame, train_frac=0.7, val_frac=0.1\n) -> Tuple[List[str], List[str], List[str]]:\n    # compute case start time\n    starts = df.groupby(\"case_id\")[\"timestamp\"].min().sort_values()\n    n = len(starts)\n    train_end = int(n * train_frac)\n    val_end = int(n * (train_frac + val_frac))\n    train_cases = starts.index[:train_end].tolist()\n    val_cases = starts.index[train_end:val_end].tolist()\n    test_cases = starts.index[val_end:].tolist()\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(\n        self,\n        df: pd.DataFrame,\n        cases: List[str],\n        scaler: StandardScaler = None,\n        fit_scaler=False,\n        max_len=10,\n    ):\n        self.samples = df[df[\"case_id\"].isin(cases)].reset_index(drop=True)\n        self.max_len = max_len\n        aux = np.stack(self.samples[\"aux\"].values).astype(np.float32)\n        if scaler is None:\n            scaler = StandardScaler()\n        if fit_scaler:\n            scaler.fit(aux)\n        self.scaler = scaler\n        aux_norm = scaler.transform(aux).astype(np.float32)\n        self.samples[\"aux_norm\"] = list(aux_norm)\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        row = self.samples.iloc[idx]\n        seq = row[\"seq\"]\n        L = min(len(seq), self.max_len)\n        padded = np.zeros(self.max_len, dtype=np.int64)\n        padded[-L:] = np.array(seq[-L:], dtype=np.int64)\n        return {\n            \"seq\": torch.tensor(padded, dtype=torch.long),\n            \"len\": torch.tensor(L, dtype=torch.long),\n            \"aux\": torch.tensor(np.array(row[\"aux_norm\"], dtype=np.float32)),\n            \"y\": torch.tensor(row[\"y\"], dtype=torch.long),\n        }\n\n\ndef collate_fn(batch):\n    collated = {}\n    keys = batch[0].keys()\n    for k in keys:\n        collated[k] = torch.stack([b[k] for b in batch], dim=0)\n    return collated\n\n\n# ------------------ Model ------------------ #\nclass NextActLSTM(nn.Module):\n    def __init__(\n        self,\n        vocab_size: int,\n        emb_dim: int = 32,\n        hidden: int = 64,\n        aux_dim: int = 5,\n        num_classes: int = 0,\n        dropout: float = 0.1,\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size, emb_dim, padding_idx=0)\n        self.lstm = nn.LSTM(emb_dim, hidden, batch_first=True)\n        self.dropout = nn.Dropout(dropout)\n        self.fc = nn.Linear(hidden + aux_dim, num_classes)\n\n    def forward(self, seq, lengths, aux):\n        emb = self.emb(seq)  # (B,L,E)\n        packed = nn.utils.rnn.pack_padded_sequence(\n            emb, lengths.cpu(), batch_first=True, enforce_sorted=False\n        )\n        out_packed, (h_n, c_n) = self.lstm(packed)\n        h_last = h_n[-1]  # (B,H)\n        x = torch.cat([h_last, aux], dim=1)\n        x = self.dropout(x)\n        logits = self.fc(x)\n        return logits\n\n\n# ------------------ Metrics ------------------ #\ndef topk_accuracy(logits, y, k=3):\n    topk = logits.topk(k, dim=1).indices\n    correct = (topk == y.unsqueeze(1)).any(dim=1).float().mean().item()\n    return correct\n\n\ndef eval_epoch(model, loader):\n    model.eval()\n    all_y = []\n    all_pred = []\n    all_logits = []\n    loss_sum = 0.0\n    n = 0\n    ce = nn.CrossEntropyLoss()\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"seq\"], batch[\"len\"], batch[\"aux\"])\n            loss = ce(logits, batch[\"y\"])\n            loss_sum += loss.item() * batch[\"y\"].size(0)\n            n += batch[\"y\"].size(0)\n            preds = torch.argmax(logits, dim=1)\n            all_y.append(batch[\"y\"].cpu().numpy())\n            all_pred.append(preds.cpu().numpy())\n            all_logits.append(logits.cpu().numpy())\n    y_true = np.concatenate(all_y) if all_y else np.array([])\n    y_pred = np.concatenate(all_pred) if all_pred else np.array([])\n    logits_all = (\n        np.concatenate(all_logits)\n        if all_logits\n        else np.zeros((0, model.fc.out_features))\n    )\n    acc = accuracy_score(y_true, y_pred) if len(y_true) > 0 else 0.0\n    macro_f1 = (\n        f1_score(y_true, y_pred, average=\"macro\", zero_division=0)\n        if len(y_true) > 0\n        else 0.0\n    )\n    # top-3\n    if len(y_true) > 0 and logits_all.shape[0] > 0:\n        top3 = []\n        for i in range(logits_all.shape[0]):\n            top3.append(int(y_true[i] in np.argsort(-logits_all[i])[:3]))\n        top3_acc = float(np.mean(top3)) if top3 else 0.0\n    else:\n        top3_acc = 0.0\n    return {\n        \"loss\": loss_sum / max(n, 1),\n        \"accuracy\": acc,\n        \"macro_f1\": macro_f1,\n        \"top3_acc\": top3_acc,\n        \"y_true\": y_true,\n        \"y_pred\": y_pred,\n        \"logits\": logits_all,\n    }\n\n\n# ------------------ Training pipeline ------------------ #\ndef run_experiment():\n    ds_name, df = load_default_dataset()\n    print(\n        f\"[data] Dataset: {ds_name}, events={len(df)}, cases={df['case_id'].nunique()}\"\n    )\n    max_len = 10\n    prefixes, act2ix = build_prefix_samples(df, max_len=max_len)\n    print(f\"[data] Prefix samples: {len(prefixes)}, vocab={len(act2ix)}\")\n    if len(prefixes) == 0 or len(act2ix) < 2:\n        print(\"[warn] Not enough data to train.\")\n        return\n    train_cases, val_cases, test_cases = time_based_split(df)\n    scaler = StandardScaler()\n    train_ds = PrefixDataset(\n        prefixes, train_cases, scaler=scaler, fit_scaler=True, max_len=max_len\n    )\n    val_ds = PrefixDataset(\n        prefixes, val_cases, scaler=scaler, fit_scaler=False, max_len=max_len\n    )\n    test_ds = PrefixDataset(\n        prefixes, test_cases, scaler=scaler, fit_scaler=False, max_len=max_len\n    )\n    print(f\"[split] train={len(train_ds)}, val={len(val_ds)}, test={len(test_ds)}\")\n    batch_size = 256\n    train_loader = DataLoader(\n        train_ds, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n    )\n    val_loader = DataLoader(\n        val_ds, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    test_loader = DataLoader(\n        test_ds, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    vocab_size = len(act2ix) + 1\n    num_classes = (\n        len(act2ix) + 1\n    )  # including PAD label index; will not appear in targets\n    model = NextActLSTM(\n        vocab_size=vocab_size,\n        emb_dim=32,\n        hidden=64,\n        aux_dim=5,\n        num_classes=num_classes,\n        dropout=0.2,\n    ).to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n    ce = nn.CrossEntropyLoss()\n    epochs = 8\n    exp_key = \"DEFAULT\"\n    for epoch in range(1, epochs + 1):\n        model.train()\n        train_loss = 0.0\n        n_seen = 0\n        for batch in train_loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"seq\"], batch[\"len\"], batch[\"aux\"])\n            loss = ce(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            train_loss += loss.item() * batch[\"y\"].size(0)\n            n_seen += batch[\"y\"].size(0)\n        train_loss /= max(n_seen, 1)\n        # Eval train subset metrics quickly using val loader approach (full train eval can be heavy; we sample a batch)\n        train_metrics = eval_epoch(model, train_loader)\n        val_metrics = eval_epoch(model, val_loader)\n        print(f\"Epoch {epoch}: validation_loss = {val_metrics['loss']:.4f}\")\n        # Track metrics\n        experiment_data[exp_key][\"epochs\"].append(epoch)\n        experiment_data[exp_key][\"losses\"][\"train\"].append(\n            {\"epoch\": epoch, \"loss\": train_loss}\n        )\n        experiment_data[exp_key][\"losses\"][\"val\"].append(\n            {\"epoch\": epoch, \"loss\": val_metrics[\"loss\"]}\n        )\n        experiment_data[exp_key][\"metrics\"][\"train\"].append(\n            {\n                \"epoch\": epoch,\n                \"accuracy\": train_metrics[\"accuracy\"],\n                \"macro_f1\": train_metrics[\"macro_f1\"],\n                \"top3_acc\": train_metrics[\"top3_acc\"],\n            }\n        )\n        experiment_data[exp_key][\"metrics\"][\"val\"].append(\n            {\n                \"epoch\": epoch,\n                \"accuracy\": val_metrics[\"accuracy\"],\n                \"macro_f1\": val_metrics[\"macro_f1\"],\n                \"top3_acc\": val_metrics[\"top3_acc\"],\n            }\n        )\n        experiment_data[exp_key][\"top3_history\"][\"train\"].append(\n            train_metrics[\"top3_acc\"]\n        )\n        experiment_data[exp_key][\"top3_history\"][\"val\"].append(val_metrics[\"top3_acc\"])\n    # Final test evaluation\n    test_metrics = eval_epoch(model, test_loader)\n    experiment_data[exp_key][\"metrics\"][\"test\"].append(\n        {\n            \"epoch\": epochs,\n            \"accuracy\": test_metrics[\"accuracy\"],\n            \"macro_f1\": test_metrics[\"macro_f1\"],\n            \"top3_acc\": test_metrics[\"top3_acc\"],\n        }\n    )\n    experiment_data[exp_key][\"predictions\"] = test_metrics[\"y_pred\"]\n    experiment_data[exp_key][\"ground_truth\"] = test_metrics[\"y_true\"]\n    # Print metrics per split\n    last_train = experiment_data[exp_key][\"metrics\"][\"train\"][-1]\n    last_val = experiment_data[exp_key][\"metrics\"][\"val\"][-1]\n    print(\n        f\"[results][{ds_name}] Train: Acc={last_train['accuracy']:.4f} MacroF1={last_train['macro_f1']:.4f} Top-3={last_train['top3_acc']:.4f}\"\n    )\n    print(\n        f\"[results][{ds_name}] Val:   Acc={last_val['accuracy']:.4f} MacroF1={last_val['macro_f1']:.4f} Top-3={last_val['top3_acc']:.4f}\"\n    )\n    print(\n        f\"[results][{ds_name}] Test:  Acc={test_metrics['accuracy']:.4f} MacroF1={test_metrics['macro_f1']:.4f} Top-3={test_metrics['top3_acc']:.4f}\"\n    )\n    # Save plots\n    try:\n        epochs_arr = experiment_data[exp_key][\"epochs\"]\n        tr_losses = [x[\"loss\"] for x in experiment_data[exp_key][\"losses\"][\"train\"]]\n        vl_losses = [x[\"loss\"] for x in experiment_data[exp_key][\"losses\"][\"val\"]]\n        plt.figure()\n        plt.plot(epochs_arr, tr_losses, label=\"train_loss\")\n        plt.plot(epochs_arr, vl_losses, label=\"val_loss\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {ds_name}\")\n        plt.savefig(os.path.join(working_dir, f\"loss_curves_{ds_name}.png\"), dpi=150)\n        plt.close()\n        tr_top3 = experiment_data[exp_key][\"top3_history\"][\"train\"]\n        vl_top3 = experiment_data[exp_key][\"top3_history\"][\"val\"]\n        plt.figure()\n        plt.plot(epochs_arr, tr_top3, label=\"train_top3\")\n        plt.plot(epochs_arr, vl_top3, label=\"val_top3\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Accuracy\")\n        plt.legend()\n        plt.title(f\"Top-3 Acc - {ds_name}\")\n        plt.savefig(os.path.join(working_dir, f\"top3_curves_{ds_name}.png\"), dpi=150)\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Also save compressed arrays for predictions\n    np.savez_compressed(\n        os.path.join(working_dir, f\"preds_{ds_name}.npz\"),\n        y_true=experiment_data[exp_key][\"ground_truth\"],\n        y_pred=experiment_data[exp_key][\"predictions\"],\n    )\n    # Print the requested metric prominently\n    print(\n        f\"Top-3 Next-Activity Accuracy (TEST) [{ds_name}]: {test_metrics['top3_acc']:.4f}\"\n    )\n\n\n# Execute\nrun_experiment()\n","plan":"We implement a simple but complete predictive process monitoring baseline for next-activity prediction using local XES logs (BPI 2012 preferred). We load the XES via pm4py, build case traces, and enforce a strict time-based split at the case level (train/val/test by case start time). We construct prefix-based samples (prefix length 1..max_len-1) and predict the next activity, using features: last-k activities (tokenized), and simple temporal features (time since start, time since last, hour, weekday, working-time flag), properly normalized on training data only. The model is a minimal 1-layer LSTM over activities with an embedding; the last hidden state is concatenated with normalized temporal features and fed to a softmax classifier. Training uses cross-entropy with early-stopping-like small epoch count, and we track and print validation loss each epoch. We evaluate Accuracy, Macro-F1, and the required Top-3 Next-Activity Accuracy on train/val each epoch and report final test metrics, saving metrics, losses, predictions, and ground truth to experiment_data.npy. We also save simple plots (loss and Top-3) per dataset under ./working. If no XES is discovered, we synthesize a small consistent event log to ensure the pipeline runs end-to-end. All tensors and the model are moved to GPU if available, and DataLoader batches are moved to device in the loop.","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\n\nos.makedirs(working_dir, exist_ok=True)\n\nimport warnings\n\nwarnings.filterwarnings(\"ignore\")\nimport pandas as pd\nfrom pathlib import Path\nfrom collections import defaultdict\nfrom typing import List, Dict, Tuple\nfrom datetime import timedelta\nimport torch\nimport torch.nn as nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score, confusion_matrix\nfrom sklearn.preprocessing import StandardScaler\n\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data container\nexperiment_data = {\n    \"DEFAULT\": {\n        \"name\": \"\",\n        \"task\": \"next-activity\",\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n        \"top3_history\": {\"train\": [], \"val\": []},\n    }\n}\n\n\ndef _has_xes(dirpath: Path) -> bool:\n    try:\n        return dirpath.is_dir() and (\n            any(dirpath.glob(\"*.xes\")) or any(dirpath.glob(\"*.xes.gz\"))\n        )\n    except Exception:\n        return False\n\n\ndef _resolve_data_dir() -> Path:\n    candidates: List[Path] = []\n    candidates += [Path(\"input\").resolve(), (Path.cwd() / \"input\").resolve()]\n    cwd = Path.cwd().resolve()\n    for base in [cwd, *cwd.parents]:\n        candidates.append((base / \"data\").resolve())\n        candidates.append((base / \"input\").resolve())\n    candidates += [\n        Path(\"/workspace/input\"),\n        Path(\"/workspace/data\"),\n        Path(\"/workspace/ai_scientist/data\"),\n        Path(\"/workspace/AI-Scientist-v2/data\"),\n        Path(\"/workspace/experiments/data\"),\n        Path(\"/workspace/ai_scientist/input\"),\n        Path(\"/workspace/experiments/input\"),\n    ]\n    seen = set()\n    for p in candidates:\n        if p in seen:\n            continue\n        seen.add(p)\n        if _has_xes(p):\n            print(f\"[data] Using discovered data dir: {p}\")\n            return p\n    raise FileNotFoundError(\"No .xes found\")\n\n\ndef xes_to_df(xes_path: Path) -> pd.DataFrame:\n    try:\n        from pm4py.objects.log.importer.xes import importer as xes_importer\n    except Exception as e:\n        raise ImportError(\"pm4py is required. pip install pm4py\") from e\n    print(f\"[data] Loading XES: {xes_path}\")\n    log = xes_importer.apply(str(xes_path))\n    rows = []\n    for tr in log:\n        case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n            \"case:concept:name\"\n        )\n        for e in tr:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": e.get(\"concept:name\"),\n                    \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                    \"timestamp\": e.get(\"time:timestamp\"),\n                    \"resource\": e.get(\"org:resource\", \"System\"),\n                }\n            )\n    df = pd.DataFrame(rows)\n    df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n    df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n    df = df.sort_values([\"timestamp\", \"case_id\"]).reset_index(drop=True)\n    return df\n\n\ndef load_default_dataset() -> Tuple[str, pd.DataFrame]:\n    datasets = {}\n    try:\n        d = _resolve_data_dir()\n        files = list(d.glob(\"*.xes\")) + list(d.glob(\"*.xes.gz\"))\n        pref = [\n            \"BPI_Challenge_2017\",\n            \"BPI2017\",\n            \"2017\",\n            \"BPI_Challenge_2012\",\n            \"BPI2012\",\n            \"2012\",\n            \"Road_Traffic_Fine_Management_Process\",\n            \"Traffic\",\n            \"Fine\",\n            \"Road\",\n        ]\n        chosen = None\n        for p in pref:\n            for f in files:\n                if p in f.name:\n                    chosen = f\n                    break\n            if chosen is not None:\n                break\n        if chosen is None and files:\n            chosen = files[0]\n        if chosen is not None:\n            df = xes_to_df(chosen)\n            name = (\n                \"BPI2017\"\n                if \"2017\" in chosen.name\n                else (\"BPI2012\" if \"2012\" in chosen.name else \"ROAD\")\n            )\n            return name, df\n    except Exception as e:\n        print(f\"[warn] Data discovery failed: {e}\")\n    print(\"[data] Generating synthetic log\")\n    rng = np.random.RandomState(42)\n    activities = [\"A\", \"B\", \"C\", \"D\", \"E\", \"F\"]\n    resources = [\"R1\", \"R2\", \"R3\"]\n    rows = []\n    start = pd.Timestamp(\"2020-01-01\", tz=\"UTC\")\n    n_cases = 200\n    for c in range(n_cases):\n        case_id = f\"C{c:04d}\"\n        t = start + pd.Timedelta(days=int(c / 5))\n        length = rng.randint(4, 9)\n        seq = [\"A\"]\n        for i in range(length - 1):\n            if seq[-1] == \"A\":\n                nxt = \"B\" if rng.rand() < 0.6 else \"C\"\n            elif seq[-1] in [\"B\", \"C\"]:\n                nxt = \"D\" if rng.rand() < 0.5 else \"E\"\n            elif seq[-1] in [\"D\", \"E\"]:\n                nxt = \"F\" if rng.rand() < 0.7 else rng.choice([\"B\", \"C\"])\n            else:\n                nxt = rng.choice(activities)\n            seq.append(nxt)\n        for act in seq:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": act,\n                    \"lifecycle\": \"complete\",\n                    \"timestamp\": t,\n                    \"resource\": rng.choice(resources),\n                }\n            )\n            t += pd.Timedelta(minutes=int(rng.exponential(120)))\n    df = pd.DataFrame(rows).sort_values([\"timestamp\", \"case_id\"]).reset_index(drop=True)\n    return \"SYNTH\", df\n\n\ndef build_prefix_samples(\n    df: pd.DataFrame, max_len: int = 10\n) -> Tuple[pd.DataFrame, Dict[str, int]]:\n    g = df.sort_values(\"timestamp\").groupby(\"case_id\")\n    traces = []\n    for cid, grp in g:\n        acts = grp[\"activity\"].tolist()\n        times = grp[\"timestamp\"].tolist()\n        res = (\n            grp[\"resource\"].tolist()\n            if \"resource\" in grp.columns\n            else [\"System\"] * len(acts)\n        )\n        traces.append((cid, acts, times, res))\n    act_set = sorted({a for _, acts, _, _ in traces for a in acts})\n    act2ix = {a: i + 1 for i, a in enumerate(act_set)}  # 0 PAD\n    samples = []\n    for cid, acts, times, res in traces:\n        if len(acts) < 2:\n            continue\n        for i in range(1, min(len(acts), max_len + 1)):\n            if i >= len(acts):\n                break\n            prefix_acts = acts[:i]\n            next_act = acts[i]\n            prefix_times = times[:i]\n            start_t = prefix_times[0]\n            last_t = prefix_times[-1]\n            delta_start = (last_t - start_t).total_seconds() / 3600.0\n            delta_last = (\n                ((prefix_times[-1] - prefix_times[-2]).total_seconds() / 3600.0)\n                if len(prefix_times) >= 2\n                else 0.0\n            )\n            hour = last_t.hour\n            weekday = last_t.weekday()\n            working = 1.0 if (weekday < 5 and 8 <= hour < 18) else 0.0\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq\": [act2ix[a] for a in prefix_acts][-max_len:],\n                    \"len\": min(len(prefix_acts), max_len),\n                    \"aux\": np.array(\n                        [delta_start, delta_last, hour, weekday, working],\n                        dtype=np.float32,\n                    ),\n                    \"y\": act2ix[next_act],\n                }\n            )\n    data = pd.DataFrame(samples)\n    return data, act2ix\n\n\ndef time_based_split(\n    df: pd.DataFrame, train_frac=0.7, val_frac=0.1\n) -> Tuple[List[str], List[str], List[str]]:\n    starts = df.groupby(\"case_id\")[\"timestamp\"].min().sort_values()\n    n = len(starts)\n    train_end = int(n * train_frac)\n    val_end = int(n * (train_frac + val_frac))\n    return (\n        starts.index[:train_end].tolist(),\n        starts.index[train_end:val_end].tolist(),\n        starts.index[val_end:].tolist(),\n    )\n\n\nclass PrefixDataset(Dataset):\n    def __init__(\n        self,\n        df: pd.DataFrame,\n        cases: List[str],\n        scaler: StandardScaler = None,\n        fit_scaler=False,\n        max_len=10,\n    ):\n        self.samples = df[df[\"case_id\"].isin(cases)].reset_index(drop=True)\n        self.max_len = max_len\n        aux = np.stack(self.samples[\"aux\"].values).astype(np.float32)\n        if scaler is None:\n            scaler = StandardScaler()\n        if fit_scaler:\n            scaler.fit(aux)\n        self.scaler = scaler\n        aux_norm = scaler.transform(aux).astype(np.float32)\n        self.samples[\"aux_norm\"] = list(aux_norm)\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        row = self.samples.iloc[idx]\n        seq = row[\"seq\"]\n        L = min(len(seq), self.max_len)\n        padded = np.zeros(self.max_len, dtype=np.int64)\n        padded[-L:] = np.array(seq[-L:], dtype=np.int64)\n        return {\n            \"seq\": torch.tensor(padded, dtype=torch.long),\n            \"len\": torch.tensor(L, dtype=torch.long),\n            \"aux\": torch.tensor(np.array(row[\"aux_norm\"], dtype=np.float32)),\n            \"y\": torch.tensor(row[\"y\"], dtype=torch.long),\n        }\n\n\ndef collate_fn(batch):\n    return {k: torch.stack([b[k] for b in batch], dim=0) for k in batch[0].keys()}\n\n\nclass NextActLSTM(nn.Module):\n    def __init__(\n        self,\n        vocab_size: int,\n        emb_dim: int = 32,\n        hidden: int = 64,\n        aux_dim: int = 5,\n        num_classes: int = 0,\n        dropout: float = 0.1,\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size, emb_dim, padding_idx=0)\n        self.lstm = nn.LSTM(emb_dim, hidden, batch_first=True)\n        self.dropout = nn.Dropout(dropout)\n        self.fc = nn.Linear(hidden + aux_dim, num_classes)\n\n    def forward(self, seq, lengths, aux):\n        emb = self.emb(seq)\n        packed = nn.utils.rnn.pack_padded_sequence(\n            emb, lengths.cpu(), batch_first=True, enforce_sorted=False\n        )\n        _, (h_n, _) = self.lstm(packed)\n        h_last = h_n[-1]\n        x = self.dropout(torch.cat([h_last, aux], dim=1))\n        return self.fc(x)\n\n\ndef eval_epoch(model, loader):\n    model.eval()\n    all_y, all_pred, all_logits = [], [], []\n    loss_sum, n = 0.0, 0\n    ce = nn.CrossEntropyLoss()\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"seq\"], batch[\"len\"], batch[\"aux\"])\n            loss = ce(logits, batch[\"y\"])\n            loss_sum += loss.item() * batch[\"y\"].size(0)\n            n += batch[\"y\"].size(0)\n            preds = torch.argmax(logits, dim=1)\n            all_y.append(batch[\"y\"].cpu().numpy())\n            all_pred.append(preds.cpu().numpy())\n            all_logits.append(logits.cpu().numpy())\n    y_true = np.concatenate(all_y) if all_y else np.array([])\n    y_pred = np.concatenate(all_pred) if all_pred else np.array([])\n    logits_all = (\n        np.concatenate(all_logits)\n        if all_logits\n        else np.zeros((0, model.fc.out_features))\n    )\n    acc = accuracy_score(y_true, y_pred) if len(y_true) > 0 else 0.0\n    macro_f1 = (\n        f1_score(y_true, y_pred, average=\"macro\", zero_division=0)\n        if len(y_true) > 0\n        else 0.0\n    )\n    if len(y_true) > 0 and logits_all.shape[0] > 0:\n        top3 = [\n            int(y_true[i] in np.argsort(-logits_all[i])[:3])\n            for i in range(logits_all.shape[0])\n        ]\n        top3_acc = float(np.mean(top3)) if top3 else 0.0\n    else:\n        top3_acc = 0.0\n    return {\n        \"loss\": loss_sum / max(n, 1),\n        \"accuracy\": acc,\n        \"macro_f1\": macro_f1,\n        \"top3_acc\": top3_acc,\n        \"y_true\": y_true,\n        \"y_pred\": y_pred,\n        \"logits\": logits_all,\n    }\n\n\ndef run_experiment():\n    ds_name, df = load_default_dataset()\n    print(\n        f\"[data] Dataset: {ds_name}, events={len(df)}, cases={df['case_id'].nunique()}\"\n    )\n    max_len = 10\n    prefixes, act2ix = build_prefix_samples(df, max_len=max_len)\n    print(f\"[data] Prefix samples: {len(prefixes)}, vocab={len(act2ix)}\")\n    if len(prefixes) == 0 or len(act2ix) < 2:\n        print(\"[warn] Not enough data to train.\")\n        return ds_name\n    train_cases, val_cases, test_cases = time_based_split(df)\n    scaler = StandardScaler()\n    train_ds = PrefixDataset(\n        prefixes, train_cases, scaler=scaler, fit_scaler=True, max_len=max_len\n    )\n    val_ds = PrefixDataset(\n        prefixes, val_cases, scaler=scaler, fit_scaler=False, max_len=max_len\n    )\n    test_ds = PrefixDataset(\n        prefixes, test_cases, scaler=scaler, fit_scaler=False, max_len=max_len\n    )\n    print(f\"[split] train={len(train_ds)}, val={len(val_ds)}, test={len(test_ds)}\")\n    train_loader = DataLoader(\n        train_ds, batch_size=256, shuffle=True, collate_fn=collate_fn\n    )\n    val_loader = DataLoader(\n        val_ds, batch_size=256, shuffle=False, collate_fn=collate_fn\n    )\n    test_loader = DataLoader(\n        test_ds, batch_size=256, shuffle=False, collate_fn=collate_fn\n    )\n    vocab_size = len(act2ix) + 1\n    num_classes = len(act2ix) + 1\n    model = NextActLSTM(\n        vocab_size=vocab_size,\n        emb_dim=32,\n        hidden=64,\n        aux_dim=5,\n        num_classes=num_classes,\n        dropout=0.2,\n    ).to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n    ce = nn.CrossEntropyLoss()\n    epochs = 8\n    exp_key = \"DEFAULT\"\n    experiment_data[exp_key][\"name\"] = ds_name\n    for epoch in range(1, epochs + 1):\n        model.train()\n        train_loss = 0.0\n        n_seen = 0\n        for batch in train_loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"seq\"], batch[\"len\"], batch[\"aux\"])\n            loss = ce(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            train_loss += loss.item() * batch[\"y\"].size(0)\n            n_seen += batch[\"y\"].size(0)\n        train_loss /= max(n_seen, 1)\n        train_metrics = eval_epoch(model, train_loader)\n        val_metrics = eval_epoch(model, val_loader)\n        print(f\"Epoch {epoch}: validation_loss = {val_metrics['loss']:.4f}\")\n        experiment_data[exp_key][\"epochs\"].append(epoch)\n        experiment_data[exp_key][\"losses\"][\"train\"].append(\n            {\"epoch\": epoch, \"loss\": train_loss}\n        )\n        experiment_data[exp_key][\"losses\"][\"val\"].append(\n            {\"epoch\": epoch, \"loss\": val_metrics[\"loss\"]}\n        )\n        experiment_data[exp_key][\"metrics\"][\"train\"].append(\n            {\n                \"epoch\": epoch,\n                \"accuracy\": train_metrics[\"accuracy\"],\n                \"macro_f1\": train_metrics[\"macro_f1\"],\n                \"top3_acc\": train_metrics[\"top3_acc\"],\n            }\n        )\n        experiment_data[exp_key][\"metrics\"][\"val\"].append(\n            {\n                \"epoch\": epoch,\n                \"accuracy\": val_metrics[\"accuracy\"],\n                \"macro_f1\": val_metrics[\"macro_f1\"],\n                \"top3_acc\": val_metrics[\"top3_acc\"],\n            }\n        )\n        experiment_data[exp_key][\"top3_history\"][\"train\"].append(\n            train_metrics[\"top3_acc\"]\n        )\n        experiment_data[exp_key][\"top3_history\"][\"val\"].append(val_metrics[\"top3_acc\"])\n    test_metrics = eval_epoch(model, test_loader)\n    experiment_data[exp_key][\"metrics\"][\"test\"].append(\n        {\n            \"epoch\": epochs,\n            \"accuracy\": test_metrics[\"accuracy\"],\n            \"macro_f1\": test_metrics[\"macro_f1\"],\n            \"top3_acc\": test_metrics[\"top3_acc\"],\n        }\n    )\n    experiment_data[exp_key][\"predictions\"] = test_metrics[\"y_pred\"].astype(int)\n    experiment_data[exp_key][\"ground_truth\"] = test_metrics[\"y_true\"].astype(int)\n    last_train = experiment_data[exp_key][\"metrics\"][\"train\"][-1]\n    last_val = experiment_data[exp_key][\"metrics\"][\"val\"][-1]\n    print(\n        f\"[results][{ds_name}] Train: Acc={last_train['accuracy']:.4f} MacroF1={last_train['macro_f1']:.4f} Top-3={last_train['top3_acc']:.4f}\"\n    )\n    print(\n        f\"[results][{ds_name}] Val:   Acc={last_val['accuracy']:.4f} MacroF1={last_val['macro_f1']:.4f} Top-3={last_val['top3_acc']:.4f}\"\n    )\n    print(\n        f\"[results][{ds_name}] Test:  Acc={test_metrics['accuracy']:.4f} MacroF1={test_metrics['macro_f1']:.4f} Top-3={test_metrics['top3_acc']:.4f}\"\n    )\n    # Save curves immediately\n    try:\n        epochs_arr = experiment_data[exp_key][\"epochs\"]\n        tr_losses = [x[\"loss\"] for x in experiment_data[exp_key][\"losses\"][\"train\"]]\n        vl_losses = [x[\"loss\"] for x in experiment_data[exp_key][\"losses\"][\"val\"]]\n        plt.figure()\n        plt.plot(epochs_arr, tr_losses, label=\"train_loss\")\n        plt.plot(epochs_arr, vl_losses, label=\"val_loss\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {ds_name} (Next-Activity)\")\n        plt.savefig(\n            os.path.join(working_dir, f\"{ds_name}_nextact_loss_curves.png\"), dpi=150\n        )\n        plt.close()\n        tr_top3 = experiment_data[exp_key][\"top3_history\"][\"train\"]\n        vl_top3 = experiment_data[exp_key][\"top3_history\"][\"val\"]\n        plt.figure()\n        plt.plot(epochs_arr, tr_top3, label=\"train_top3\")\n        plt.plot(epochs_arr, vl_top3, label=\"val_top3\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Accuracy\")\n        plt.legend()\n        plt.title(f\"Top-3 Accuracy Curves - {ds_name} (Next-Activity)\")\n        plt.savefig(\n            os.path.join(working_dir, f\"{ds_name}_nextact_top3_curves.png\"), dpi=150\n        )\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n        plt.close()\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    np.savez_compressed(\n        os.path.join(working_dir, f\"preds_{ds_name}.npz\"),\n        y_true=experiment_data[exp_key][\"ground_truth\"],\n        y_pred=experiment_data[exp_key][\"predictions\"],\n    )\n    print(\n        f\"Top-3 Next-Activity Accuracy (TEST) [{ds_name}]: {test_metrics['top3_acc']:.4f}\"\n    )\n    # Return final metrics for printing\n    return ds_name, test_metrics\n\n\n# Execute training + evaluation\nds_name, test_metrics = run_experiment()\n\n# Load experiment_data and create standard visualizations (only from available data)\ntry:\n    experiment_data = np.load(\n        os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n    ).item()\nexcept Exception as e:\n    print(f\"Error loading experiment data: {e}\")\n    experiment_data = {}\n\n# Plot 1: Training/Validation Loss Curves\ntry:\n    exp_key = \"DEFAULT\"\n    ds = experiment_data.get(exp_key, {})\n    name = ds.get(\"name\", \"DATASET\")\n    epochs_arr = ds.get(\"epochs\", [])\n    if (\n        epochs_arr\n        and ds.get(\"losses\", {}).get(\"train\")\n        and ds.get(\"losses\", {}).get(\"val\")\n    ):\n        tr_losses = [x[\"loss\"] for x in ds[\"losses\"][\"train\"]]\n        vl_losses = [x[\"loss\"] for x in ds[\"losses\"][\"val\"]]\n        plt.figure()\n        plt.plot(epochs_arr, tr_losses, label=\"Train Loss\")\n        plt.plot(epochs_arr, vl_losses, label=\"Val Loss\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.legend()\n        plt.title(f\"{name} - Next-Activity: Loss Curves\")\n        plt.suptitle(\"Left: Train, Right: N/A (single view)\")\n        plt.savefig(\n            os.path.join(working_dir, f\"{name}_nextact_loss_curves_from_expdata.png\"),\n            dpi=150,\n        )\n        plt.close()\nexcept Exception as e:\n    print(f\"Error creating plot1: {e}\")\n    plt.close()\n\n# Plot 2: Top-3 Accuracy Curves\ntry:\n    exp_key = \"DEFAULT\"\n    ds = experiment_data.get(exp_key, {})\n    name = ds.get(\"name\", \"DATASET\")\n    epochs_arr = ds.get(\"epochs\", [])\n    tr_top3 = ds.get(\"top3_history\", {}).get(\"train\", [])\n    vl_top3 = ds.get(\"top3_history\", {}).get(\"val\", [])\n    if epochs_arr and tr_top3 and vl_top3:\n        plt.figure()\n        plt.plot(epochs_arr, tr_top3, label=\"Train Top-3\")\n        plt.plot(epochs_arr, vl_top3, label=\"Val Top-3\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Accuracy\")\n        plt.legend()\n        plt.title(f\"{name} - Next-Activity: Top-3 Accuracy Curves\")\n        plt.suptitle(\"Left: Train, Right: Validation\")\n        plt.savefig(\n            os.path.join(working_dir, f\"{name}_nextact_top3_curves_from_expdata.png\"),\n            dpi=150,\n        )\n        plt.close()\nexcept Exception as e:\n    print(f\"Error creating plot2: {e}\")\n    plt.close()\n\n# Plot 3: Test Confusion Matrix\ntry:\n    exp_key = \"DEFAULT\"\n    ds = experiment_data.get(exp_key, {})\n    name = ds.get(\"name\", \"DATASET\")\n    y_true = np.array(ds.get(\"ground_truth\", []))\n    y_pred = np.array(ds.get(\"predictions\", []))\n    if y_true.size > 0 and y_pred.size > 0:\n        cm = confusion_matrix(y_true, y_pred)\n        plt.figure(figsize=(6, 5))\n        im = plt.imshow(cm, cmap=\"Blues\")\n        plt.colorbar(im)\n        plt.title(f\"{name} - Next-Activity: Confusion Matrix (Test)\")\n        plt.suptitle(\"Left: Ground Truth, Right: Predicted (Indices)\")\n        plt.xlabel(\"Predicted\")\n        plt.ylabel(\"True\")\n        ticks = np.arange(cm.shape[0])\n        if len(ticks) <= 30:\n            plt.xticks(ticks)\n            plt.yticks(ticks)\n        plt.tight_layout()\n        plt.savefig(\n            os.path.join(working_dir, f\"{name}_nextact_confusion_matrix_test.png\"),\n            dpi=150,\n        )\n        plt.close()\nexcept Exception as e:\n    print(f\"Error creating plot3: {e}\")\n    plt.close()\n\n# Print evaluation metrics prominently\ntry:\n    print(f\"TEST Accuracy: {test_metrics['accuracy']:.4f}\")\n    print(f\"TEST Macro-F1: {test_metrics['macro_f1']:.4f}\")\n    print(f\"TEST Top-3 Accuracy: {test_metrics['top3_acc']:.4f}\")\nexcept Exception as e:\n    print(f\"Error printing metrics: {e}\")\n","plot_plan":"We will load a local XES event log using pm4py, convert it to a dataframe, and build next-activity prefix samples with simple temporal auxiliary features. Cases will be split by their start timestamp to enforce a proper time-based train/val/test division. We will train a lightweight LSTM with embeddings over activity sequences plus auxiliary features to predict the next activity. During training, we will track accuracy, macro-F1, and Top-3 accuracy, along with loss curves for train/validation splits. After training, we will evaluate on the test set and print the BPM metrics (accuracy, macro-F1, Top-3). We will save an experiment_data.npy file that contains metrics, predictions, and ground truth for plotting. Then, we will load experiment_data.npy to generate standard visualizations: training/validation loss curves, top-3 curves, and a test confusion matrix for next-activity. Plots will be saved to the working directory, each in its own try-except block, only using data that exists in experiment_data.npy.","step":3,"id":"ea3c544524e64e3fbd193a8593b37724","ctime":1757756700.3383868,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<4:41:43,  1.86it/s]","\rparsing log, completed traces ::   0%|          | 70/31509 [00:00<03:35, 145.84it/s]","\rparsing log, completed traces ::   0%|          | 132/31509 [00:00<02:03, 254.61it/s]","\rparsing log, completed traces ::   1%|          | 182/31509 [00:01<02:48, 185.99it/s]","\rparsing log, completed traces ::   1%|          | 245/31509 [00:01<01:59, 262.47it/s]","\rparsing log, completed traces ::   1%|          | 311/31509 [00:01<01:31, 340.90it/s]","\rparsing log, completed traces ::   1%|1         | 377/31509 [00:01<01:15, 410.92it/s]","\rparsing log, completed traces ::   1%|1         | 441/31509 [00:01<01:07, 463.41it/s]","\rparsing log, completed traces ::   2%|1         | 508/31509 [00:01<01:00, 513.64it/s]","\rparsing log, completed traces ::   2%|1         | 569/31509 [00:01<00:57, 537.45it/s]","\rparsing log, completed traces ::   2%|2         | 640/31509 [00:01<00:53, 582.33it/s]","\rparsing log, completed traces ::   2%|2         | 714/31509 [00:01<00:49, 626.38it/s]","\rparsing log, completed traces ::   2%|2         | 781/31509 [00:02<00:48, 628.81it/s]","\rparsing log, completed traces ::   3%|2         | 847/31509 [00:02<00:48, 636.78it/s]","\rparsing log, completed traces ::   3%|2         | 913/31509 [00:02<00:47, 642.45it/s]","\rparsing log, completed traces ::   3%|3         | 986/31509 [00:02<00:45, 667.46it/s]","\rparsing log, completed traces ::   3%|3         | 1054/31509 [00:02<00:46, 656.46it/s]","\rparsing log, completed traces ::   4%|3         | 1121/31509 [00:02<01:18, 386.05it/s]","\rparsing log, completed traces ::   4%|3         | 1194/31509 [00:02<01:06, 453.23it/s]","\rparsing log, completed traces ::   4%|4         | 1265/31509 [00:02<00:59, 508.81it/s]","\rparsing log, completed traces ::   4%|4         | 1330/31509 [00:03<00:55, 542.20it/s]","\rparsing log, completed traces ::   4%|4         | 1400/31509 [00:03<00:51, 582.15it/s]","\rparsing log, completed traces ::   5%|4         | 1466/31509 [00:03<00:50, 597.14it/s]","\rparsing log, completed traces ::   5%|4         | 1534/31509 [00:03<00:48, 618.81it/s]","\rparsing log, completed traces ::   5%|5         | 1600/31509 [00:03<00:47, 628.88it/s]","\rparsing log, completed traces ::   5%|5         | 1666/31509 [00:03<00:47, 630.96it/s]","\rparsing log, completed traces ::   6%|5         | 1737/31509 [00:03<00:45, 653.20it/s]","\rparsing log, completed traces ::   6%|5         | 1804/31509 [00:03<00:46, 634.54it/s]","\rparsing log, completed traces ::   6%|5         | 1869/31509 [00:03<00:47, 622.70it/s]","\rparsing log, completed traces ::   6%|6         | 1937/31509 [00:04<00:46, 638.82it/s]","\rparsing log, completed traces ::   6%|6         | 2005/31509 [00:04<00:45, 650.16it/s]","\rparsing log, completed traces ::   7%|6         | 2071/31509 [00:04<00:45, 651.22it/s]","\rparsing log, completed traces ::   7%|6         | 2139/31509 [00:04<00:44, 659.06it/s]","\rparsing log, completed traces ::   7%|7         | 2206/31509 [00:04<01:19, 366.44it/s]","\rparsing log, completed traces ::   7%|7         | 2279/31509 [00:04<01:07, 435.09it/s]","\rparsing log, completed traces ::   7%|7         | 2345/31509 [00:04<01:00, 482.34it/s]","\rparsing log, completed traces ::   8%|7         | 2414/31509 [00:04<00:54, 530.10it/s]","\rparsing log, completed traces ::   8%|7         | 2482/31509 [00:05<00:51, 566.32it/s]","\rparsing log, completed traces ::   8%|8         | 2551/31509 [00:05<00:48, 597.33it/s]","\rparsing log, completed traces ::   8%|8         | 2626/31509 [00:05<00:45, 637.78it/s]","\rparsing log, completed traces ::   9%|8         | 2695/31509 [00:05<00:44, 651.02it/s]","\rparsing log, completed traces ::   9%|8         | 2764/31509 [00:05<00:44, 648.98it/s]","\rparsing log, completed traces ::   9%|8         | 2834/31509 [00:05<00:43, 663.25it/s]","\rparsing log, completed traces ::   9%|9         | 2904/31509 [00:05<00:42, 669.70it/s]","\rparsing log, completed traces ::   9%|9         | 2978/31509 [00:05<00:41, 685.93it/s]","\rparsing log, completed traces ::  10%|9         | 3048/31509 [00:05<00:41, 685.57it/s]","\rparsing log, completed traces ::  10%|9         | 3118/31509 [00:06<00:43, 659.70it/s]","\rparsing log, completed traces ::  10%|#         | 3188/31509 [00:06<00:42, 666.55it/s]","\rparsing log, completed traces ::  10%|#         | 3256/31509 [00:06<00:42, 667.39it/s]","\rparsing log, completed traces ::  11%|#         | 3332/31509 [00:06<00:40, 693.83it/s]","\rparsing log, completed traces ::  11%|#         | 3407/31509 [00:06<00:39, 707.89it/s]","\rparsing log, completed traces ::  11%|#1        | 3479/31509 [00:06<01:16, 367.67it/s]","\rparsing log, completed traces ::  11%|#1        | 3547/31509 [00:06<01:06, 422.67it/s]","\rparsing log, completed traces ::  11%|#1        | 3620/31509 [00:07<00:57, 485.01it/s]","\rparsing log, completed traces ::  12%|#1        | 3692/31509 [00:07<00:51, 536.36it/s]","\rparsing log, completed traces ::  12%|#1        | 3760/31509 [00:07<00:48, 570.83it/s]","\rparsing log, completed traces ::  12%|#2        | 3827/31509 [00:07<00:46, 596.33it/s]","\rparsing log, completed traces ::  12%|#2        | 3898/31509 [00:07<00:44, 626.22it/s]","\rparsing log, completed traces ::  13%|#2        | 3969/31509 [00:07<00:42, 649.29it/s]","\rparsing log, completed traces ::  13%|#2        | 4042/31509 [00:07<00:41, 668.95it/s]","\rparsing log, completed traces ::  13%|#3        | 4114/31509 [00:07<00:40, 683.33it/s]","\rparsing log, completed traces ::  13%|#3        | 4185/31509 [00:07<00:40, 676.34it/s]","\rparsing log, completed traces ::  14%|#3        | 4255/31509 [00:07<00:41, 663.43it/s]","\rparsing log, completed traces ::  14%|#3        | 4323/31509 [00:08<00:40, 664.37it/s]","\rparsing log, completed traces ::  14%|#3        | 4393/31509 [00:08<00:40, 672.95it/s]","\rparsing log, completed traces ::  14%|#4        | 4461/31509 [00:08<00:40, 670.99it/s]","\rparsing log, completed traces ::  14%|#4        | 4532/31509 [00:08<00:39, 682.17it/s]","\rparsing log, completed traces ::  15%|#4        | 4603/31509 [00:08<00:39, 689.31it/s]","\rparsing log, completed traces ::  15%|#4        | 4674/31509 [00:08<00:38, 693.05it/s]","\rparsing log, completed traces ::  15%|#5        | 4749/31509 [00:08<00:37, 708.80it/s]","\rparsing log, completed traces ::  15%|#5        | 4820/31509 [00:08<00:38, 697.95it/s]","\rparsing log, completed traces ::  16%|#5        | 4890/31509 [00:09<01:18, 337.37it/s]","\rparsing log, completed traces ::  16%|#5        | 4955/31509 [00:09<01:08, 389.10it/s]","\rparsing log, completed traces ::  16%|#5        | 5022/31509 [00:09<00:59, 442.91it/s]","\rparsing log, completed traces ::  16%|#6        | 5094/31509 [00:09<00:52, 501.63it/s]","\rparsing log, completed traces ::  16%|#6        | 5161/31509 [00:09<00:48, 540.35it/s]","\rparsing log, completed traces ::  17%|#6        | 5232/31509 [00:09<00:45, 580.90it/s]","\rparsing log, completed traces ::  17%|#6        | 5302/31509 [00:09<00:42, 610.28it/s]","\rparsing log, completed traces ::  17%|#7        | 5370/31509 [00:09<00:41, 627.47it/s]","\rparsing log, completed traces ::  17%|#7        | 5437/31509 [00:10<00:41, 633.44it/s]","\rparsing log, completed traces ::  17%|#7        | 5511/31509 [00:10<00:39, 662.80it/s]","\rparsing log, completed traces ::  18%|#7        | 5583/31509 [00:10<00:38, 678.58it/s]","\rparsing log, completed traces ::  18%|#7        | 5655/31509 [00:10<00:37, 690.60it/s]","\rparsing log, completed traces ::  18%|#8        | 5732/31509 [00:10<00:36, 712.99it/s]","\rparsing log, completed traces ::  18%|#8        | 5805/31509 [00:10<00:36, 708.99it/s]","\rparsing log, completed traces ::  19%|#8        | 5880/31509 [00:10<00:35, 718.47it/s]","\rparsing log, completed traces ::  19%|#8        | 5956/31509 [00:10<00:35, 729.22it/s]","\rparsing log, completed traces ::  19%|#9        | 6030/31509 [00:10<00:35, 712.36it/s]","\rparsing log, completed traces ::  19%|#9        | 6102/31509 [00:10<00:35, 705.80it/s]","\rparsing log, completed traces ::  20%|#9        | 6176/31509 [00:11<00:35, 712.61it/s]","\rparsing log, completed traces ::  20%|#9        | 6249/31509 [00:11<00:35, 713.86it/s]","\rparsing log, completed traces ::  20%|##        | 6321/31509 [00:11<00:35, 700.74it/s]","\rparsing log, completed traces ::  20%|##        | 6396/31509 [00:11<00:35, 712.51it/s]","\rparsing log, completed traces ::  21%|##        | 6468/31509 [00:11<00:35, 703.60it/s]","\rparsing log, completed traces ::  21%|##        | 6539/31509 [00:12<01:19, 312.86it/s]","\rparsing log, completed traces ::  21%|##        | 6603/31509 [00:12<01:08, 363.50it/s]","\rparsing log, completed traces ::  21%|##1       | 6674/31509 [00:12<00:58, 426.06it/s]","\rparsing log, completed traces ::  21%|##1       | 6746/31509 [00:12<00:51, 485.17it/s]","\rparsing log, completed traces ::  22%|##1       | 6816/31509 [00:12<00:46, 532.47it/s]","\rparsing log, completed traces ::  22%|##1       | 6884/31509 [00:12<00:43, 567.51it/s]","\rparsing log, completed traces ::  22%|##2       | 6952/31509 [00:12<00:41, 595.94it/s]","\rparsing log, completed traces ::  22%|##2       | 7029/31509 [00:12<00:38, 641.12it/s]","\rparsing log, completed traces ::  23%|##2       | 7102/31509 [00:12<00:36, 662.96it/s]","\rparsing log, completed traces ::  23%|##2       | 7173/31509 [00:12<00:36, 668.71it/s]","\rparsing log, completed traces ::  23%|##2       | 7243/31509 [00:13<00:36, 671.49it/s]","\rparsing log, completed traces ::  23%|##3       | 7313/31509 [00:13<00:36, 671.51it/s]","\rparsing log, completed traces ::  23%|##3       | 7382/31509 [00:13<00:36, 669.86it/s]","\rparsing log, completed traces ::  24%|##3       | 7450/31509 [00:13<00:35, 669.94it/s]","\rparsing log, completed traces ::  24%|##3       | 7528/31509 [00:13<00:34, 701.74it/s]","\rparsing log, completed traces ::  24%|##4       | 7601/31509 [00:13<00:33, 709.35it/s]","\rparsing log, completed traces ::  24%|##4       | 7673/31509 [00:13<00:34, 687.83it/s]","\rparsing log, completed traces ::  25%|##4       | 7743/31509 [00:13<00:34, 683.50it/s]","\rparsing log, completed traces ::  25%|##4       | 7812/31509 [00:13<00:35, 669.67it/s]","\rparsing log, completed traces ::  25%|##5       | 7886/31509 [00:13<00:34, 689.72it/s]","\rparsing log, completed traces ::  25%|##5       | 7956/31509 [00:14<00:34, 689.64it/s]","\rparsing log, completed traces ::  25%|##5       | 8028/31509 [00:14<00:33, 698.06it/s]","\rparsing log, completed traces ::  26%|##5       | 8098/31509 [00:14<00:33, 694.21it/s]","\rparsing log, completed traces ::  26%|##5       | 8168/31509 [00:14<00:33, 694.40it/s]","\rparsing log, completed traces ::  26%|##6       | 8243/31509 [00:14<00:32, 710.28it/s]","\rparsing log, completed traces ::  26%|##6       | 8315/31509 [00:15<01:13, 314.40it/s]","\rparsing log, completed traces ::  27%|##6       | 8381/31509 [00:15<01:02, 368.35it/s]","\rparsing log, completed traces ::  27%|##6       | 8449/31509 [00:15<00:54, 424.47it/s]","\rparsing log, completed traces ::  27%|##7       | 8518/31509 [00:15<00:47, 479.46it/s]","\rparsing log, completed traces ::  27%|##7       | 8583/31509 [00:15<00:44, 517.56it/s]","\rparsing log, completed traces ::  27%|##7       | 8650/31509 [00:15<00:41, 555.00it/s]","\rparsing log, completed traces ::  28%|##7       | 8720/31509 [00:15<00:38, 592.60it/s]","\rparsing log, completed traces ::  28%|##7       | 8790/31509 [00:15<00:36, 621.18it/s]","\rparsing log, completed traces ::  28%|##8       | 8858/31509 [00:15<00:35, 634.47it/s]","\rparsing log, completed traces ::  28%|##8       | 8926/31509 [00:15<00:35, 639.97it/s]","\rparsing log, completed traces ::  29%|##8       | 8994/31509 [00:16<00:34, 649.05it/s]","\rparsing log, completed traces ::  29%|##8       | 9061/31509 [00:16<00:35, 634.96it/s]","\rparsing log, completed traces ::  29%|##8       | 9129/31509 [00:16<00:34, 646.58it/s]","\rparsing log, completed traces ::  29%|##9       | 9197/31509 [00:16<00:34, 655.11it/s]","\rparsing log, completed traces ::  29%|##9       | 9264/31509 [00:16<00:34, 640.47it/s]","\rparsing log, completed traces ::  30%|##9       | 9338/31509 [00:16<00:33, 667.67it/s]","\rparsing log, completed traces ::  30%|##9       | 9406/31509 [00:16<00:33, 657.81it/s]","\rparsing log, completed traces ::  30%|###       | 9474/31509 [00:16<00:33, 661.27it/s]","\rparsing log, completed traces ::  30%|###       | 9541/31509 [00:16<00:33, 661.78it/s]","\rparsing log, completed traces ::  30%|###       | 9608/31509 [00:16<00:33, 656.18it/s]","\rparsing log, completed traces ::  31%|###       | 9675/31509 [00:17<00:33, 657.91it/s]","\rparsing log, completed traces ::  31%|###       | 9741/31509 [00:17<00:33, 649.58it/s]","\rparsing log, completed traces ::  31%|###1      | 9807/31509 [00:17<00:33, 652.62it/s]","\rparsing log, completed traces ::  31%|###1      | 9873/31509 [00:17<00:33, 649.78it/s]","\rparsing log, completed traces ::  32%|###1      | 9939/31509 [00:17<00:33, 647.29it/s]","\rparsing log, completed traces ::  32%|###1      | 10004/31509 [00:17<00:33, 639.89it/s]","\rparsing log, completed traces ::  32%|###1      | 10069/31509 [00:17<00:33, 636.59it/s]","\rparsing log, completed traces ::  32%|###2      | 10133/31509 [00:17<00:34, 623.13it/s]","\rparsing log, completed traces ::  32%|###2      | 10196/31509 [00:18<01:21, 261.36it/s]","\rparsing log, completed traces ::  33%|###2      | 10266/31509 [00:18<01:05, 325.90it/s]","\rparsing log, completed traces ::  33%|###2      | 10326/31509 [00:18<00:57, 371.03it/s]","\rparsing log, completed traces ::  33%|###2      | 10390/31509 [00:18<00:49, 423.83it/s]","\rparsing log, completed traces ::  33%|###3      | 10454/31509 [00:18<00:44, 471.12it/s]","\rparsing log, completed traces ::  33%|###3      | 10516/31509 [00:18<00:41, 505.13it/s]","\rparsing log, completed traces ::  34%|###3      | 10583/31509 [00:18<00:38, 545.53it/s]","\rparsing log, completed traces ::  34%|###3      | 10645/31509 [00:19<00:37, 562.87it/s]","\rparsing log, completed traces ::  34%|###3      | 10709/31509 [00:19<00:35, 579.29it/s]","\rparsing log, completed traces ::  34%|###4      | 10772/31509 [00:19<00:35, 589.04it/s]","\rparsing log, completed traces ::  34%|###4      | 10834/31509 [00:19<00:35, 587.69it/s]","\rparsing log, completed traces ::  35%|###4      | 10895/31509 [00:19<00:35, 586.38it/s]","\rparsing log, completed traces ::  35%|###4      | 10956/31509 [00:19<00:34, 591.74it/s]","\rparsing log, completed traces ::  35%|###4      | 11026/31509 [00:19<00:33, 620.66it/s]","\rparsing log, completed traces ::  35%|###5      | 11092/31509 [00:19<00:32, 631.40it/s]","\rparsing log, completed traces ::  35%|###5      | 11158/31509 [00:19<00:31, 639.59it/s]","\rparsing log, completed traces ::  36%|###5      | 11225/31509 [00:19<00:31, 648.17it/s]","\rparsing log, completed traces ::  36%|###5      | 11295/31509 [00:20<00:30, 662.33it/s]","\rparsing log, completed traces ::  36%|###6      | 11362/31509 [00:20<00:31, 647.75it/s]","\rparsing log, completed traces ::  36%|###6      | 11427/31509 [00:20<00:31, 641.31it/s]","\rparsing log, completed traces ::  36%|###6      | 11492/31509 [00:20<00:31, 637.70it/s]","\rparsing log, completed traces ::  37%|###6      | 11556/31509 [00:20<00:31, 629.20it/s]","\rparsing log, completed traces ::  37%|###6      | 11621/31509 [00:20<00:31, 634.96it/s]","\rparsing log, completed traces ::  37%|###7      | 11685/31509 [00:20<00:31, 634.45it/s]","\rparsing log, completed traces ::  37%|###7      | 11754/31509 [00:20<00:30, 649.21it/s]","\rparsing log, completed traces ::  38%|###7      | 11822/31509 [00:20<00:29, 658.26it/s]","\rparsing log, completed traces ::  38%|###7      | 11888/31509 [00:21<00:29, 654.87it/s]","\rparsing log, completed traces ::  38%|###7      | 11960/31509 [00:21<00:29, 672.22it/s]","\rparsing log, completed traces ::  38%|###8      | 12028/31509 [00:21<00:29, 662.48it/s]","\rparsing log, completed traces ::  38%|###8      | 12095/31509 [00:21<00:29, 662.99it/s]","\rparsing log, completed traces ::  39%|###8      | 12162/31509 [00:21<00:29, 660.80it/s]","\rparsing log, completed traces ::  39%|###8      | 12233/31509 [00:21<00:28, 673.41it/s]","\rparsing log, completed traces ::  39%|###9      | 12301/31509 [00:22<01:15, 256.10it/s]","\rparsing log, completed traces ::  39%|###9      | 12369/31509 [00:22<01:00, 314.70it/s]","\rparsing log, completed traces ::  39%|###9      | 12433/31509 [00:22<00:51, 368.16it/s]","\rparsing log, completed traces ::  40%|###9      | 12500/31509 [00:22<00:44, 425.15it/s]","\rparsing log, completed traces ::  40%|###9      | 12564/31509 [00:22<00:40, 470.61it/s]","\rparsing log, completed traces ::  40%|####      | 12634/31509 [00:22<00:36, 523.28it/s]","\rparsing log, completed traces ::  40%|####      | 12699/31509 [00:22<00:33, 554.53it/s]","\rparsing log, completed traces ::  41%|####      | 12767/31509 [00:22<00:31, 587.28it/s]","\rparsing log, completed traces ::  41%|####      | 12836/31509 [00:22<00:30, 611.23it/s]","\rparsing log, completed traces ::  41%|####      | 12902/31509 [00:23<00:29, 620.52it/s]","\rparsing log, completed traces ::  41%|####1     | 12968/31509 [00:23<00:29, 630.06it/s]","\rparsing log, completed traces ::  41%|####1     | 13038/31509 [00:23<00:28, 649.84it/s]","\rparsing log, completed traces ::  42%|####1     | 13113/31509 [00:23<00:27, 678.47it/s]","\rparsing log, completed traces ::  42%|####1     | 13183/31509 [00:23<00:27, 659.12it/s]","\rparsing log, completed traces ::  42%|####2     | 13251/31509 [00:23<00:27, 663.06it/s]","\rparsing log, completed traces ::  42%|####2     | 13319/31509 [00:23<00:27, 661.83it/s]","\rparsing log, completed traces ::  42%|####2     | 13387/31509 [00:23<00:27, 664.67it/s]","\rparsing log, completed traces ::  43%|####2     | 13454/31509 [00:23<00:27, 661.69it/s]","\rparsing log, completed traces ::  43%|####2     | 13521/31509 [00:24<00:27, 661.54it/s]","\rparsing log, completed traces ::  43%|####3     | 13588/31509 [00:24<00:27, 652.84it/s]","\rparsing log, completed traces ::  43%|####3     | 13654/31509 [00:24<00:27, 644.90it/s]","\rparsing log, completed traces ::  44%|####3     | 13719/31509 [00:24<00:27, 642.05it/s]","\rparsing log, completed traces ::  44%|####3     | 13784/31509 [00:24<00:27, 635.05it/s]","\rparsing log, completed traces ::  44%|####3     | 13852/31509 [00:24<00:27, 644.92it/s]","\rparsing log, completed traces ::  44%|####4     | 13917/31509 [00:24<00:27, 643.95it/s]","\rparsing log, completed traces ::  44%|####4     | 13982/31509 [00:24<00:27, 645.24it/s]","\rparsing log, completed traces ::  45%|####4     | 14053/31509 [00:24<00:26, 664.14it/s]","\rparsing log, completed traces ::  45%|####4     | 14121/31509 [00:24<00:26, 667.28it/s]","\rparsing log, completed traces ::  45%|####5     | 14190/31509 [00:25<00:25, 672.65it/s]","\rparsing log, completed traces ::  45%|####5     | 14258/31509 [00:25<00:25, 673.73it/s]","\rparsing log, completed traces ::  45%|####5     | 14327/31509 [00:25<00:25, 677.88it/s]","\rparsing log, completed traces ::  46%|####5     | 14401/31509 [00:25<00:24, 694.54it/s]","\rparsing log, completed traces ::  46%|####5     | 14471/31509 [00:25<00:24, 691.37it/s]","\rparsing log, completed traces ::  46%|####6     | 14541/31509 [00:25<00:25, 664.59it/s]","\rparsing log, completed traces ::  46%|####6     | 14608/31509 [00:25<00:25, 657.71it/s]","\rparsing log, completed traces ::  47%|####6     | 14676/31509 [00:25<00:25, 661.64it/s]","\rparsing log, completed traces ::  47%|####6     | 14743/31509 [00:26<01:08, 243.89it/s]","\rparsing log, completed traces ::  47%|####7     | 14816/31509 [00:26<00:54, 307.15it/s]","\rparsing log, completed traces ::  47%|####7     | 14883/31509 [00:26<00:45, 364.13it/s]","\rparsing log, completed traces ::  47%|####7     | 14959/31509 [00:26<00:37, 437.04it/s]","\rparsing log, completed traces ::  48%|####7     | 15024/31509 [00:26<00:34, 481.08it/s]","\rparsing log, completed traces ::  48%|####7     | 15099/31509 [00:26<00:30, 541.86it/s]","\rparsing log, completed traces ::  48%|####8     | 15167/31509 [00:27<00:28, 572.17it/s]","\rparsing log, completed traces ::  48%|####8     | 15236/31509 [00:27<00:27, 602.40it/s]","\rparsing log, completed traces ::  49%|####8     | 15310/31509 [00:27<00:25, 636.94it/s]","\rparsing log, completed traces ::  49%|####8     | 15384/31509 [00:27<00:24, 662.77it/s]","\rparsing log, completed traces ::  49%|####9     | 15457/31509 [00:27<00:23, 681.35it/s]","\rparsing log, completed traces ::  49%|####9     | 15529/31509 [00:27<00:23, 688.82it/s]","\rparsing log, completed traces ::  50%|####9     | 15600/31509 [00:27<00:22, 692.64it/s]","\rparsing log, completed traces ::  50%|####9     | 15676/31509 [00:27<00:22, 706.18it/s]","\rparsing log, completed traces ::  50%|####9     | 15748/31509 [00:27<00:22, 694.32it/s]","\rparsing log, completed traces ::  50%|#####     | 15819/31509 [00:27<00:22, 691.67it/s]","\rparsing log, completed traces ::  50%|#####     | 15895/31509 [00:28<00:22, 707.66it/s]","\rparsing log, completed traces ::  51%|#####     | 15967/31509 [00:28<00:21, 706.52it/s]","\rparsing log, completed traces ::  51%|#####     | 16038/31509 [00:28<00:22, 696.67it/s]","\rparsing log, completed traces ::  51%|#####1    | 16114/31509 [00:28<00:21, 713.60it/s]","\rparsing log, completed traces ::  51%|#####1    | 16191/31509 [00:28<00:21, 727.11it/s]","\rparsing log, completed traces ::  52%|#####1    | 16264/31509 [00:28<00:21, 719.23it/s]","\rparsing log, completed traces ::  52%|#####1    | 16337/31509 [00:28<00:21, 714.37it/s]","\rparsing log, completed traces ::  52%|#####2    | 16409/31509 [00:28<00:21, 688.66it/s]","\rparsing log, completed traces ::  52%|#####2    | 16479/31509 [00:28<00:21, 686.14it/s]","\rparsing log, completed traces ::  53%|#####2    | 16548/31509 [00:29<00:21, 680.73it/s]","\rparsing log, completed traces ::  53%|#####2    | 16619/31509 [00:29<00:21, 687.42it/s]","\rparsing log, completed traces ::  53%|#####2    | 16695/31509 [00:29<00:20, 705.77it/s]","\rparsing log, completed traces ::  53%|#####3    | 16766/31509 [00:29<00:21, 696.59it/s]","\rparsing log, completed traces ::  53%|#####3    | 16837/31509 [00:29<00:20, 699.72it/s]","\rparsing log, completed traces ::  54%|#####3    | 16908/31509 [00:29<00:20, 701.81it/s]","\rparsing log, completed traces ::  54%|#####3    | 16979/31509 [00:29<00:20, 702.65it/s]","\rparsing log, completed traces ::  54%|#####4    | 17063/31509 [00:29<00:19, 739.90it/s]","\rparsing log, completed traces ::  54%|#####4    | 17138/31509 [00:29<00:19, 723.71it/s]","\rparsing log, completed traces ::  55%|#####4    | 17211/31509 [00:29<00:20, 706.59it/s]","\rparsing log, completed traces ::  55%|#####4    | 17287/31509 [00:30<00:19, 721.85it/s]","\rparsing log, completed traces ::  55%|#####5    | 17360/31509 [00:30<00:19, 711.79it/s]","\rparsing log, completed traces ::  55%|#####5    | 17438/31509 [00:30<00:19, 730.80it/s]","\rparsing log, completed traces ::  56%|#####5    | 17512/31509 [00:30<00:20, 697.48it/s]","\rparsing log, completed traces ::  56%|#####5    | 17587/31509 [00:30<00:19, 712.33it/s]","\rparsing log, completed traces ::  56%|#####6    | 17659/31509 [00:31<00:58, 237.10it/s]","\rparsing log, completed traces ::  56%|#####6    | 17730/31509 [00:31<00:46, 293.89it/s]","\rparsing log, completed traces ::  57%|#####6    | 17803/31509 [00:31<00:38, 357.52it/s]","\rparsing log, completed traces ::  57%|#####6    | 17878/31509 [00:31<00:32, 424.46it/s]","\rparsing log, completed traces ::  57%|#####6    | 17953/31509 [00:31<00:27, 487.36it/s]","\rparsing log, completed traces ::  57%|#####7    | 18027/31509 [00:31<00:24, 542.42it/s]","\rparsing log, completed traces ::  57%|#####7    | 18103/31509 [00:31<00:22, 592.53it/s]","\rparsing log, completed traces ::  58%|#####7    | 18177/31509 [00:31<00:21, 627.62it/s]","\rparsing log, completed traces ::  58%|#####7    | 18249/31509 [00:32<00:20, 637.45it/s]","\rparsing log, completed traces ::  58%|#####8    | 18320/31509 [00:32<00:20, 637.47it/s]","\rparsing log, completed traces ::  58%|#####8    | 18397/31509 [00:32<00:19, 671.44it/s]","\rparsing log, completed traces ::  59%|#####8    | 18468/31509 [00:32<00:19, 680.56it/s]","\rparsing log, completed traces ::  59%|#####8    | 18539/31509 [00:32<00:19, 677.48it/s]","\rparsing log, completed traces ::  59%|#####9    | 18609/31509 [00:32<00:19, 674.31it/s]","\rparsing log, completed traces ::  59%|#####9    | 18688/31509 [00:32<00:18, 706.85it/s]","\rparsing log, completed traces ::  60%|#####9    | 18762/31509 [00:32<00:17, 713.73it/s]","\rparsing log, completed traces ::  60%|#####9    | 18835/31509 [00:32<00:17, 707.78it/s]","\rparsing log, completed traces ::  60%|######    | 18914/31509 [00:33<00:17, 728.97it/s]","\rparsing log, completed traces ::  60%|######    | 18988/31509 [00:33<00:17, 713.30it/s]","\rparsing log, completed traces ::  60%|######    | 19060/31509 [00:33<00:17, 711.76it/s]","\rparsing log, completed traces ::  61%|######    | 19136/31509 [00:33<00:17, 724.03it/s]","\rparsing log, completed traces ::  61%|######    | 19209/31509 [00:33<00:17, 713.53it/s]","\rparsing log, completed traces ::  61%|######1   | 19281/31509 [00:33<00:17, 705.17it/s]","\rparsing log, completed traces ::  61%|######1   | 19353/31509 [00:33<00:17, 707.25it/s]","\rparsing log, completed traces ::  62%|######1   | 19432/31509 [00:33<00:16, 729.50it/s]","\rparsing log, completed traces ::  62%|######1   | 19506/31509 [00:33<00:16, 722.01it/s]","\rparsing log, completed traces ::  62%|######2   | 19580/31509 [00:33<00:16, 724.62it/s]","\rparsing log, completed traces ::  62%|######2   | 19653/31509 [00:34<00:16, 718.12it/s]","\rparsing log, completed traces ::  63%|######2   | 19725/31509 [00:34<00:16, 710.04it/s]","\rparsing log, completed traces ::  63%|######2   | 19797/31509 [00:34<00:17, 686.16it/s]","\rparsing log, completed traces ::  63%|######3   | 19866/31509 [00:34<00:16, 685.95it/s]","\rparsing log, completed traces ::  63%|######3   | 19935/31509 [00:34<00:16, 685.82it/s]","\rparsing log, completed traces ::  63%|######3   | 20004/31509 [00:34<00:16, 678.23it/s]","\rparsing log, completed traces ::  64%|######3   | 20073/31509 [00:34<00:16, 679.52it/s]","\rparsing log, completed traces ::  64%|######3   | 20142/31509 [00:34<00:16, 680.65it/s]","\rparsing log, completed traces ::  64%|######4   | 20213/31509 [00:34<00:16, 686.70it/s]","\rparsing log, completed traces ::  64%|######4   | 20289/31509 [00:34<00:15, 707.39it/s]","\rparsing log, completed traces ::  65%|######4   | 20360/31509 [00:35<00:16, 690.49it/s]","\rparsing log, completed traces ::  65%|######4   | 20441/31509 [00:35<00:15, 722.53it/s]","\rparsing log, completed traces ::  65%|######5   | 20514/31509 [00:35<00:15, 719.03it/s]","\rparsing log, completed traces ::  65%|######5   | 20590/31509 [00:35<00:15, 727.48it/s]","\rparsing log, completed traces ::  66%|######5   | 20668/31509 [00:35<00:14, 741.84it/s]","\rparsing log, completed traces ::  66%|######5   | 20744/31509 [00:35<00:14, 742.23it/s]","\rparsing log, completed traces ::  66%|######6   | 20819/31509 [00:35<00:14, 738.38it/s]","\rparsing log, completed traces ::  66%|######6   | 20893/31509 [00:35<00:14, 724.16it/s]","\rparsing log, completed traces ::  67%|######6   | 20966/31509 [00:36<00:48, 217.43it/s]","\rparsing log, completed traces ::  67%|######6   | 21034/31509 [00:36<00:38, 268.72it/s]","\rparsing log, completed traces ::  67%|######6   | 21102/31509 [00:36<00:32, 324.80it/s]","\rparsing log, completed traces ::  67%|######7   | 21170/31509 [00:37<00:27, 382.75it/s]","\rparsing log, completed traces ::  67%|######7   | 21244/31509 [00:37<00:22, 450.34it/s]","\rparsing log, completed traces ::  68%|######7   | 21316/31509 [00:37<00:20, 507.56it/s]","\rparsing log, completed traces ::  68%|######7   | 21384/31509 [00:37<00:18, 540.67it/s]","\rparsing log, completed traces ::  68%|######8   | 21459/31509 [00:37<00:16, 592.28it/s]","\rparsing log, completed traces ::  68%|######8   | 21532/31509 [00:37<00:15, 626.31it/s]","\rparsing log, completed traces ::  69%|######8   | 21604/31509 [00:37<00:15, 650.82it/s]","\rparsing log, completed traces ::  69%|######8   | 21675/31509 [00:37<00:15, 642.45it/s]","\rparsing log, completed traces ::  69%|######9   | 21754/31509 [00:37<00:14, 682.73it/s]","\rparsing log, completed traces ::  69%|######9   | 21826/31509 [00:37<00:14, 689.18it/s]","\rparsing log, completed traces ::  70%|######9   | 21902/31509 [00:38<00:13, 709.42it/s]","\rparsing log, completed traces ::  70%|######9   | 21975/31509 [00:38<00:13, 700.81it/s]","\rparsing log, completed traces ::  70%|######9   | 22052/31509 [00:38<00:13, 720.68it/s]","\rparsing log, completed traces ::  70%|#######   | 22125/31509 [00:38<00:13, 714.98it/s]","\rparsing log, completed traces ::  70%|#######   | 22198/31509 [00:38<00:12, 718.47it/s]","\rparsing log, completed traces ::  71%|#######   | 22271/31509 [00:38<00:13, 708.35it/s]","\rparsing log, completed traces ::  71%|#######   | 22343/31509 [00:38<00:12, 709.54it/s]","\rparsing log, completed traces ::  71%|#######1  | 22418/31509 [00:38<00:12, 719.44it/s]","\rparsing log, completed traces ::  71%|#######1  | 22494/31509 [00:38<00:12, 730.11it/s]","\rparsing log, completed traces ::  72%|#######1  | 22568/31509 [00:38<00:12, 709.83it/s]","\rparsing log, completed traces ::  72%|#######1  | 22640/31509 [00:39<00:12, 706.12it/s]","\rparsing log, completed traces ::  72%|#######2  | 22711/31509 [00:39<00:12, 701.98it/s]","\rparsing log, completed traces ::  72%|#######2  | 22786/31509 [00:39<00:12, 713.72it/s]","\rparsing log, completed traces ::  73%|#######2  | 22861/31509 [00:39<00:11, 720.86it/s]","\rparsing log, completed traces ::  73%|#######2  | 22934/31509 [00:39<00:12, 697.80it/s]","\rparsing log, completed traces ::  73%|#######3  | 23006/31509 [00:39<00:12, 703.79it/s]","\rparsing log, completed traces ::  73%|#######3  | 23077/31509 [00:39<00:11, 703.63it/s]","\rparsing log, completed traces ::  73%|#######3  | 23148/31509 [00:39<00:11, 699.60it/s]","\rparsing log, completed traces ::  74%|#######3  | 23219/31509 [00:39<00:11, 697.47it/s]","\rparsing log, completed traces ::  74%|#######3  | 23289/31509 [00:39<00:11, 690.99it/s]","\rparsing log, completed traces ::  74%|#######4  | 23359/31509 [00:40<00:12, 677.18it/s]","\rparsing log, completed traces ::  74%|#######4  | 23431/31509 [00:40<00:11, 687.04it/s]","\rparsing log, completed traces ::  75%|#######4  | 23500/31509 [00:40<00:11, 682.57it/s]","\rparsing log, completed traces ::  75%|#######4  | 23575/31509 [00:40<00:11, 697.80it/s]","\rparsing log, completed traces ::  75%|#######5  | 23647/31509 [00:40<00:11, 702.73it/s]","\rparsing log, completed traces ::  75%|#######5  | 23718/31509 [00:40<00:11, 704.24it/s]","\rparsing log, completed traces ::  76%|#######5  | 23790/31509 [00:40<00:10, 707.82it/s]","\rparsing log, completed traces ::  76%|#######5  | 23861/31509 [00:40<00:10, 705.49it/s]","\rparsing log, completed traces ::  76%|#######5  | 23932/31509 [00:40<00:10, 695.17it/s]","\rparsing log, completed traces ::  76%|#######6  | 24002/31509 [00:41<00:10, 693.57it/s]","\rparsing log, completed traces ::  76%|#######6  | 24072/31509 [00:41<00:10, 692.86it/s]","\rparsing log, completed traces ::  77%|#######6  | 24142/31509 [00:41<00:10, 689.08it/s]","\rparsing log, completed traces ::  77%|#######6  | 24212/31509 [00:41<00:10, 691.12it/s]","\rparsing log, completed traces ::  77%|#######7  | 24282/31509 [00:41<00:10, 692.39it/s]","\rparsing log, completed traces ::  77%|#######7  | 24358/31509 [00:41<00:10, 709.59it/s]","\rparsing log, completed traces ::  78%|#######7  | 24429/31509 [00:41<00:10, 678.53it/s]","\rparsing log, completed traces ::  78%|#######7  | 24500/31509 [00:41<00:10, 687.20it/s]","\rparsing log, completed traces ::  78%|#######7  | 24570/31509 [00:41<00:10, 689.00it/s]","\rparsing log, completed traces ::  78%|#######8  | 24646/31509 [00:41<00:09, 708.66it/s]","\rparsing log, completed traces ::  78%|#######8  | 24718/31509 [00:42<00:34, 196.19it/s]","\rparsing log, completed traces ::  79%|#######8  | 24789/31509 [00:43<00:26, 249.64it/s]","\rparsing log, completed traces ::  79%|#######8  | 24864/31509 [00:43<00:21, 313.92it/s]","\rparsing log, completed traces ::  79%|#######9  | 24935/31509 [00:43<00:17, 375.64it/s]","\rparsing log, completed traces ::  79%|#######9  | 25015/31509 [00:43<00:14, 451.93it/s]","\rparsing log, completed traces ::  80%|#######9  | 25085/31509 [00:43<00:13, 492.39it/s]","\rparsing log, completed traces ::  80%|#######9  | 25158/31509 [00:43<00:11, 544.86it/s]","\rparsing log, completed traces ::  80%|########  | 25230/31509 [00:43<00:10, 585.23it/s]","\rparsing log, completed traces ::  80%|########  | 25300/31509 [00:43<00:10, 604.38it/s]","\rparsing log, completed traces ::  81%|########  | 25374/31509 [00:43<00:09, 638.73it/s]","\rparsing log, completed traces ::  81%|########  | 25446/31509 [00:43<00:09, 658.83it/s]","\rparsing log, completed traces ::  81%|########  | 25517/31509 [00:44<00:08, 672.38it/s]","\rparsing log, completed traces ::  81%|########1 | 25595/31509 [00:44<00:08, 702.16it/s]","\rparsing log, completed traces ::  81%|########1 | 25668/31509 [00:44<00:08, 690.15it/s]","\rparsing log, completed traces ::  82%|########1 | 25739/31509 [00:44<00:08, 692.30it/s]","\rparsing log, completed traces ::  82%|########1 | 25810/31509 [00:44<00:08, 690.12it/s]","\rparsing log, completed traces ::  82%|########2 | 25880/31509 [00:44<00:08, 687.06it/s]","\rparsing log, completed traces ::  82%|########2 | 25950/31509 [00:44<00:08, 690.46it/s]","\rparsing log, completed traces ::  83%|########2 | 26020/31509 [00:44<00:08, 676.89it/s]","\rparsing log, completed traces ::  83%|########2 | 26089/31509 [00:44<00:08, 670.59it/s]","\rparsing log, completed traces ::  83%|########3 | 26163/31509 [00:44<00:07, 687.32it/s]","\rparsing log, completed traces ::  83%|########3 | 26233/31509 [00:45<00:07, 688.21it/s]","\rparsing log, completed traces ::  83%|########3 | 26303/31509 [00:45<00:07, 690.64it/s]","\rparsing log, completed traces ::  84%|########3 | 26378/31509 [00:45<00:07, 706.21it/s]","\rparsing log, completed traces ::  84%|########3 | 26449/31509 [00:45<00:07, 699.70it/s]","\rparsing log, completed traces ::  84%|########4 | 26520/31509 [00:45<00:07, 687.61it/s]","\rparsing log, completed traces ::  84%|########4 | 26589/31509 [00:45<00:07, 674.44it/s]","\rparsing log, completed traces ::  85%|########4 | 26659/31509 [00:45<00:07, 674.08it/s]","\rparsing log, completed traces ::  85%|########4 | 26733/31509 [00:45<00:06, 692.94it/s]","\rparsing log, completed traces ::  85%|########5 | 26803/31509 [00:45<00:06, 686.07it/s]","\rparsing log, completed traces ::  85%|########5 | 26872/31509 [00:46<00:06, 684.83it/s]","\rparsing log, completed traces ::  86%|########5 | 26942/31509 [00:46<00:06, 687.77it/s]","\rparsing log, completed traces ::  86%|########5 | 27011/31509 [00:46<00:06, 677.35it/s]","\rparsing log, completed traces ::  86%|########5 | 27079/31509 [00:46<00:06, 673.13it/s]","\rparsing log, completed traces ::  86%|########6 | 27151/31509 [00:46<00:06, 686.52it/s]","\rparsing log, completed traces ::  86%|########6 | 27223/31509 [00:46<00:06, 694.45it/s]","\rparsing log, completed traces ::  87%|########6 | 27296/31509 [00:46<00:06, 700.06it/s]","\rparsing log, completed traces ::  87%|########6 | 27367/31509 [00:46<00:06, 687.52it/s]","\rparsing log, completed traces ::  87%|########7 | 27437/31509 [00:46<00:05, 688.22it/s]","\rparsing log, completed traces ::  87%|########7 | 27506/31509 [00:46<00:05, 686.22it/s]","\rparsing log, completed traces ::  88%|########7 | 27579/31509 [00:47<00:05, 696.33it/s]","\rparsing log, completed traces ::  88%|########7 | 27651/31509 [00:47<00:05, 701.43it/s]","\rparsing log, completed traces ::  88%|########7 | 27725/31509 [00:47<00:05, 712.27it/s]","\rparsing log, completed traces ::  88%|########8 | 27797/31509 [00:47<00:05, 711.43it/s]","\rparsing log, completed traces ::  88%|########8 | 27869/31509 [00:47<00:05, 691.69it/s]","\rparsing log, completed traces ::  89%|########8 | 27939/31509 [00:47<00:05, 682.64it/s]","\rparsing log, completed traces ::  89%|########8 | 28008/31509 [00:47<00:05, 671.92it/s]","\rparsing log, completed traces ::  89%|########9 | 28076/31509 [00:47<00:05, 660.05it/s]","\rparsing log, completed traces ::  89%|########9 | 28143/31509 [00:47<00:05, 658.31it/s]","\rparsing log, completed traces ::  90%|########9 | 28209/31509 [00:47<00:05, 656.18it/s]","\rparsing log, completed traces ::  90%|########9 | 28282/31509 [00:48<00:04, 676.72it/s]","\rparsing log, completed traces ::  90%|########9 | 28350/31509 [00:48<00:04, 666.87it/s]","\rparsing log, completed traces ::  90%|######### | 28417/31509 [00:48<00:04, 657.52it/s]","\rparsing log, completed traces ::  90%|######### | 28486/31509 [00:48<00:04, 666.79it/s]","\rparsing log, completed traces ::  91%|######### | 28553/31509 [00:48<00:04, 665.79it/s]","\rparsing log, completed traces ::  91%|######### | 28624/31509 [00:48<00:04, 676.49it/s]","\rparsing log, completed traces ::  91%|#########1| 28692/31509 [00:48<00:04, 671.08it/s]","\rparsing log, completed traces ::  91%|#########1| 28760/31509 [00:49<00:15, 175.40it/s]","\rparsing log, completed traces ::  91%|#########1| 28821/31509 [00:49<00:12, 217.80it/s]","\rparsing log, completed traces ::  92%|#########1| 28884/31509 [00:49<00:09, 268.11it/s]","\rparsing log, completed traces ::  92%|#########1| 28955/31509 [00:50<00:07, 333.67it/s]","\rparsing log, completed traces ::  92%|#########2| 29025/31509 [00:50<00:06, 397.78it/s]","\rparsing log, completed traces ::  92%|#########2| 29091/31509 [00:50<00:05, 449.76it/s]","\rparsing log, completed traces ::  93%|#########2| 29161/31509 [00:50<00:04, 503.95it/s]","\rparsing log, completed traces ::  93%|#########2| 29233/31509 [00:50<00:04, 553.43it/s]","\rparsing log, completed traces ::  93%|#########2| 29301/31509 [00:50<00:03, 583.85it/s]","\rparsing log, completed traces ::  93%|#########3| 29368/31509 [00:50<00:03, 603.58it/s]","\rparsing log, completed traces ::  93%|#########3| 29435/31509 [00:50<00:03, 612.41it/s]","\rparsing log, completed traces ::  94%|#########3| 29501/31509 [00:50<00:03, 614.62it/s]","\rparsing log, completed traces ::  94%|#########3| 29566/31509 [00:50<00:03, 613.67it/s]","\rparsing log, completed traces ::  94%|#########4| 29640/31509 [00:51<00:02, 648.47it/s]","\rparsing log, completed traces ::  94%|#########4| 29713/31509 [00:51<00:02, 669.91it/s]","\rparsing log, completed traces ::  95%|#########4| 29782/31509 [00:51<00:02, 661.49it/s]","\rparsing log, completed traces ::  95%|#########4| 29850/31509 [00:51<00:02, 660.91it/s]","\rparsing log, completed traces ::  95%|#########4| 29921/31509 [00:51<00:02, 675.14it/s]","\rparsing log, completed traces ::  95%|#########5| 29991/31509 [00:51<00:02, 681.27it/s]","\rparsing log, completed traces ::  95%|#########5| 30060/31509 [00:51<00:02, 670.67it/s]","\rparsing log, completed traces ::  96%|#########5| 30134/31509 [00:51<00:01, 690.47it/s]","\rparsing log, completed traces ::  96%|#########5| 30204/31509 [00:51<00:01, 678.91it/s]","\rparsing log, completed traces ::  96%|#########6| 30277/31509 [00:52<00:01, 693.44it/s]","\rparsing log, completed traces ::  96%|#########6| 30347/31509 [00:52<00:01, 694.12it/s]","\rparsing log, completed traces ::  97%|#########6| 30419/31509 [00:52<00:01, 701.31it/s]","\rparsing log, completed traces ::  97%|#########6| 30490/31509 [00:52<00:01, 695.48it/s]","\rparsing log, completed traces ::  97%|#########6| 30563/31509 [00:52<00:01, 702.38it/s]","\rparsing log, completed traces ::  97%|#########7| 30636/31509 [00:52<00:01, 708.48it/s]","\rparsing log, completed traces ::  97%|#########7| 30708/31509 [00:52<00:01, 711.40it/s]","\rparsing log, completed traces ::  98%|#########7| 30780/31509 [00:52<00:01, 708.90it/s]","\rparsing log, completed traces ::  98%|#########7| 30852/31509 [00:52<00:00, 711.51it/s]","\rparsing log, completed traces ::  98%|#########8| 30924/31509 [00:52<00:00, 696.38it/s]","\rparsing log, completed traces ::  98%|#########8| 30994/31509 [00:53<00:00, 696.44it/s]","\rparsing log, completed traces ::  99%|#########8| 31067/31509 [00:53<00:00, 704.97it/s]","\rparsing log, completed traces ::  99%|#########8| 31141/31509 [00:53<00:00, 715.00it/s]","\rparsing log, completed traces ::  99%|#########9| 31217/31509 [00:53<00:00, 728.08it/s]","\rparsing log, completed traces ::  99%|#########9| 31290/31509 [00:53<00:00, 685.35it/s]","\rparsing log, completed traces :: 100%|#########9| 31360/31509 [00:53<00:00, 676.83it/s]","\rparsing log, completed traces :: 100%|#########9| 31433/31509 [00:53<00:00, 691.17it/s]","\rparsing log, completed traces :: 100%|#########9| 31503/31509 [00:53<00:00, 675.80it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:53<00:00, 585.79it/s]","\n","[data] Dataset: BPI2017, events=1202267, cases=31509","\n","[data] Prefix samples: 315072, vocab=26","\n","[split] train=220550, val=31507, test=63015","\n","Epoch 1: validation_loss = 0.8507","\n","Epoch 2: validation_loss = 0.7853","\n","Epoch 3: validation_loss = 0.7690","\n","Epoch 4: validation_loss = 0.7631","\n","Epoch 5: validation_loss = 0.7601","\n","Epoch 6: validation_loss = 0.7600","\n","Epoch 7: validation_loss = 0.7568","\n","Epoch 8: validation_loss = 0.7555","\n","[results][BPI2017] Train: Acc=0.6674 MacroF1=0.3715 Top-3=0.9767","\n","[results][BPI2017] Val:   Acc=0.6736 MacroF1=0.4361 Top-3=0.9824","\n","[results][BPI2017] Test:  Acc=0.6696 MacroF1=0.3637 Top-3=0.9784","\n","Top-3 Next-Activity Accuracy (TEST) [BPI2017]: 0.9784","\n","Execution time: 12 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"I will load the saved experiment data from the working directory, parse the nested structure to obtain losses and metrics for train, validation, and test splits, and compute best (min for loss, max for accuracy/F1/top-3) or final values as appropriate. The dataset names will come from the top-level keys in the saved experiment data (e.g., DEFAULT). The script will print the dataset name first, followed by clearly labeled metrics such as training loss, validation loss, training accuracy, validation accuracy, test accuracy, test macro F1 score, and test top-3 accuracy. It will handle missing sections gracefully and will not generate any plots.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef load_experiment_data(path):\n    return np.load(path, allow_pickle=True).item()\n\n\ndef safe_min(items, key):\n    if not items:\n        return None\n    best = min(items, key=lambda d: d.get(key, float(\"inf\")))\n    return best.get(key, None), best.get(\"epoch\", None)\n\n\ndef safe_max(items, key):\n    if not items:\n        return None\n    best = max(items, key=lambda d: d.get(key, float(\"-inf\")))\n    return best.get(key, None), best.get(\"epoch\", None)\n\n\ndef print_metric(label, value, epoch=None):\n    if value is None:\n        print(f\"{label}: N/A\")\n    else:\n        if isinstance(value, float):\n            # print floats with 4 decimals\n            if epoch is not None:\n                print(f\"{label}: {value:.4f} (epoch {epoch})\")\n            else:\n                print(f\"{label}: {value:.4f}\")\n        else:\n            if epoch is not None:\n                print(f\"{label}: {value} (epoch {epoch})\")\n            else:\n                print(f\"{label}: {value}\")\n\n\ndef main():\n    # 0. Working directory\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    # 1. Load experiment_data.npy from working directory\n    exp_path = os.path.join(working_dir, \"experiment_data.npy\")\n    if not os.path.exists(exp_path):\n        raise FileNotFoundError(f\"experiment_data.npy not found at {exp_path}\")\n    experiment_data = load_experiment_data(exp_path)\n\n    # 2. Extract metrics per dataset key\n    for ds_key, ds_data in experiment_data.items():\n        print(f\"Dataset: {ds_key}\")\n\n        metrics = ds_data.get(\"metrics\", {})\n        losses = ds_data.get(\"losses\", {})\n\n        train_losses = losses.get(\"train\", []) or []\n        val_losses = losses.get(\"val\", []) or []\n\n        # Best training/validation loss (minimum)\n        best_tr_loss, tr_loss_epoch = safe_min(train_losses, \"loss\")\n        best_vl_loss, vl_loss_epoch = safe_min(val_losses, \"loss\")\n\n        print_metric(\"training loss (best)\", best_tr_loss, tr_loss_epoch)\n        print_metric(\"validation loss (best)\", best_vl_loss, vl_loss_epoch)\n\n        # Train metrics (best by max)\n        tr_metrics_list = metrics.get(\"train\", []) or []\n        tr_acc, tr_acc_epoch = safe_max(tr_metrics_list, \"accuracy\")\n        tr_f1, tr_f1_epoch = safe_max(tr_metrics_list, \"macro_f1\")\n        tr_top3, tr_top3_epoch = safe_max(tr_metrics_list, \"top3_acc\")\n\n        print_metric(\"training accuracy (best)\", tr_acc, tr_acc_epoch)\n        print_metric(\"training macro F1 score (best)\", tr_f1, tr_f1_epoch)\n        print_metric(\"training top-3 accuracy (best)\", tr_top3, tr_top3_epoch)\n\n        # Validation metrics (best by max)\n        vl_metrics_list = metrics.get(\"val\", []) or []\n        vl_acc, vl_acc_epoch = safe_max(vl_metrics_list, \"accuracy\")\n        vl_f1, vl_f1_epoch = safe_max(vl_metrics_list, \"macro_f1\")\n        vl_top3, vl_top3_epoch = safe_max(vl_metrics_list, \"top3_acc\")\n\n        print_metric(\"validation accuracy (best)\", vl_acc, vl_acc_epoch)\n        print_metric(\"validation macro F1 score (best)\", vl_f1, vl_f1_epoch)\n        print_metric(\"validation top-3 accuracy (best)\", vl_top3, vl_top3_epoch)\n\n        # Test metrics (final value; list typically has one element)\n        test_metrics_list = metrics.get(\"test\", []) or []\n        if test_metrics_list:\n            test_entry = test_metrics_list[-1]\n            print_metric(\"test accuracy (final)\", test_entry.get(\"accuracy\", None))\n            print_metric(\n                \"test macro F1 score (final)\", test_entry.get(\"macro_f1\", None)\n            )\n            print_metric(\n                \"test top-3 accuracy (final)\", test_entry.get(\"top3_acc\", None)\n            )\n        else:\n            print_metric(\"test accuracy (final)\", None)\n            print_metric(\"test macro F1 score (final)\", None)\n            print_metric(\"test top-3 accuracy (final)\", None)\n\n\n# Execute immediately without if __name__ == \"__main__\":\nmain()\n","parse_term_out":["Dataset: DEFAULT","\n","training loss (best): 0.7992 (epoch 8)","\n","validation loss (best): 0.7555 (epoch 8)","\n","training accuracy (best): 0.6678 (epoch 6)","\n","training macro F1 score (best): 0.3715 (epoch 8)","\n","training top-3 accuracy (best): 0.9767 (epoch 8)","\n","validation accuracy (best): 0.6736 (epoch 8)","\n","validation macro F1 score (best): 0.4361 (epoch 8)","\n","validation top-3 accuracy (best): 0.9824 (epoch 6)","\n","test accuracy (final): 0.6696","\n","test macro F1 score (final): 0.3637","\n","test top-3 accuracy (final): 0.9784","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":721.36128282547,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"The script executed successfully end-to-end on the local BPI 2017 XES log. Data discovery and pm4py loading worked, prefix samples were built, a proper time-based case split was applied, and an LSTM baseline for next-activity prediction was trained and evaluated. Training converged (val loss decreased), and metrics were reported for train/val/test, including accuracy, macro-F1, and top-3 accuracy (Test top-3 \u2248 0.9784). No crashes or runtime errors were observed.\n\nMinor observations and potential improvements (not blocking):\n- The classifier includes a PAD class (num_classes = vocab_size = len(act2ix)+1) even though targets are 1..len(act2ix); this wastes a logit but doesn\u2019t break correctness.\n- build_prefix_samples does not filter lifecycle to \u2018complete\u2019 and may include non-complete transitions for some logs, potentially duplicating events; consider filtering to lifecycle == 'complete' if present.\n- The activity vocabulary is built on the full dataset; while this avoids OOV at test time, stricter setups build the vocab on train and map unseen to an OOV token.\n- Consider early stopping or learning rate scheduling, class weighting if label imbalance is high, and logging seeds for full reproducibility.\n\nOverall, the preliminary implementation meets the stage goals and constraints (time-based split, next-activity target, baseline LSTM, required metrics).","exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088","metric":{"value":{"metric_names":[{"metric_name":"training loss","lower_is_better":true,"description":"Average loss computed on the training split. Lower values indicate better fit.","data":[{"dataset_name":"DEFAULT","final_value":0.7992,"best_value":0.7992}]},{"metric_name":"validation loss","lower_is_better":true,"description":"Average loss computed on the validation split used for model selection. Lower values indicate better generalization.","data":[{"dataset_name":"DEFAULT","final_value":0.7555,"best_value":0.7555}]},{"metric_name":"training accuracy","lower_is_better":false,"description":"Proportion of correct predictions on the training split.","data":[{"dataset_name":"DEFAULT","final_value":0.6678,"best_value":0.6678}]},{"metric_name":"training macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the training split, averaging F1 across classes.","data":[{"dataset_name":"DEFAULT","final_value":0.3715,"best_value":0.3715}]},{"metric_name":"training top-3 accuracy","lower_is_better":false,"description":"Proportion of samples where the correct label is within the top 3 predicted classes on the training split.","data":[{"dataset_name":"DEFAULT","final_value":0.9767,"best_value":0.9767}]},{"metric_name":"validation accuracy","lower_is_better":false,"description":"Proportion of correct predictions on the validation split.","data":[{"dataset_name":"DEFAULT","final_value":0.6736,"best_value":0.6736}]},{"metric_name":"validation macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the validation split, averaging F1 across classes.","data":[{"dataset_name":"DEFAULT","final_value":0.4361,"best_value":0.4361}]},{"metric_name":"validation top-3 accuracy","lower_is_better":false,"description":"Proportion of samples where the correct label is within the top 3 predicted classes on the validation split.","data":[{"dataset_name":"DEFAULT","final_value":0.9824,"best_value":0.9824}]},{"metric_name":"test accuracy","lower_is_better":false,"description":"Proportion of correct predictions on the held-out test split.","data":[{"dataset_name":"DEFAULT","final_value":0.6696,"best_value":0.6696}]},{"metric_name":"test macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the held-out test split, averaging F1 across classes.","data":[{"dataset_name":"DEFAULT","final_value":0.3637,"best_value":0.3637}]},{"metric_name":"test top-3 accuracy","lower_is_better":false,"description":"Proportion of samples where the correct label is within the top 3 predicted classes on the held-out test split.","data":[{"dataset_name":"DEFAULT","final_value":0.9784,"best_value":0.9784}]}]},"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":false,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":["../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_top3_curves_from_expdata.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/top3_curves_BPI2017.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_loss_curves.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_top3_curves.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_confusion_matrix_test.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2012_loss.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/loss_curves_BPI2017.png","../../logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_loss_curves_from_expdata.png"],"plot_paths":["experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_top3_curves_from_expdata.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/top3_curves_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_loss_curves.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_top3_curves.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_confusion_matrix_test.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2012_loss.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/loss_curves_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_loss_curves_from_expdata.png"],"plot_analyses":[{"analysis":"The top-3 accuracy curves for both training and validation sets show a steady increase in performance over the epochs. The validation accuracy starts higher than the training accuracy, which indicates good generalization capacity of the model. The consistent improvement suggests that the model is learning effectively and there is no overfitting observed.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_top3_curves_from_expdata.png"},{"analysis":"The top-3 accuracy curves presented here are consistent with the previous plot, showing similar trends in both training and validation performance. This confirms that the model is stable and maintains its predictive performance across different runs or views.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/top3_curves_BPI2017.png"},{"analysis":"The loss curves for training and validation show a typical decline over the epochs, indicating effective learning. The validation loss being consistently lower than the training loss suggests good generalization and no signs of overfitting.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_loss_curves.png"},{"analysis":"The top-3 accuracy curves indicate a stable and high performance on both training and validation sets, similar to the earlier plots. This stability across different views points to a robust model setup.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_top3_curves.png"},{"analysis":"The confusion matrix displays a concentration of correct predictions, especially in the lower indices which likely represent the most frequent events. However, there is some dispersion in predictions for less frequent events, indicating room for improvement in handling less common activities.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_confusion_matrix_test.png"},{"analysis":"The loss curves for BPI2012 show a rapid decrease in loss for both training and validation sets, becoming stable after a few epochs. This suggests effective learning and convergence without overfitting, as indicated by the close values of training and validation loss.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2012_loss.png"},{"analysis":"These loss curves are consistent with previous observations, showing a decrease in loss over epochs for both training and validation. The validation loss remains slightly lower than the training loss, reinforcing the model's ability to generalize well.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/loss_curves_BPI2017.png"},{"analysis":"The loss curves depict a clear trend of decreasing loss over epochs, with validation loss consistently below training loss, indicating strong generalization and effective learning of the predictive model.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_ea3c544524e64e3fbd193a8593b37724_proc_332088/BPI2017_nextact_loss_curves_from_expdata.png"}],"vlm_feedback_summary":"The experimental results demonstrate effective learning and generalization capabilities of the model across various metrics like top-3 accuracy and loss reduction. The model shows consistent performance improvements over epochs without signs of overfitting. The confusion matrix analysis suggests that while frequent events are predicted accurately, there is some room for improvement in predicting less common events. Overall, the approach indicates a promising direction for resource-centric predictive process monitoring.","datasets_successfully_tested":["[BPI2012]"],"ablation_name":null,"hyperparam_name":null,"is_seed_node":false,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"import os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nimport random\nimport math\nimport matplotlib.pyplot as plt\n\n# Device handling (required)\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data structure\nexperiment_data = {\n    \"BPI2012\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"BPI2017\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"ROAD\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n}\n\n# Data loading utilities (use provided helper)\nfrom ai_scientist.ideas.my_research_topic import load_datasets, pick_default_dataset\n\n\n# Reproducibility\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# Build prefixes\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    # Keep only 'complete' transitions if lifecycle exists\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        df = df[df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")]\n        if len(df) == 0:\n            df = df.copy()  # fallback if empty\n            df = df.sort_values([\"case_id\", \"timestamp\"])\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    # Build activity vocab\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}  # 0 for PAD\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        # Convert to numpy arrays for safe positional indexing\n        ts_ns = (\n            pd.to_datetime(g[\"timestamp\"], utc=True).astype(\"int64\").to_numpy()\n        )  # nanoseconds\n        ts = (ts_ns // 10**9).astype(np.int64)  # seconds as numpy array\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str).tolist()], dtype=np.int64\n        )\n        # simple calendar features\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        # time deltas and since start in seconds\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(\n            np.float32\n        )  # [T,5]\n        T = len(acts_ids)\n        if T < 2:\n            continue\n        # Generate prefixes of length k (min_prefix_len..min(max_prefix_len, T-1)); target = activity at position k\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            seq_acts = acts_ids[:k].tolist()\n            seq_feats = feats[:k]\n            target = int(acts_ids[k])\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": seq_acts,\n                    \"seq_feats\": seq_feats.copy(),\n                    \"target\": target,\n                    \"last_ts\": int(ts[k - 1]),\n                    \"next_ts\": int(ts[k]),\n                }\n            )\n\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n\n    # Collect feature normalization stats over all feats (initial; will be recomputed on train split)\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n    ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n    for s in samples:\n        if s[\"seq_feats\"].shape[0] > 0:\n            s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n            s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n    return samples, act2id, id2act, pad_id\n\n\n# Time-based split by case start time\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.zeros((pad_len, self.num_cont), dtype=np.float32)\n        feats_pad = np.vstack([feats_pad, feats.astype(np.float32)])\n        attn_mask = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad, dtype=torch.long),\n            \"feats\": torch.tensor(feats_pad, dtype=torch.float32),\n            \"mask\": torch.tensor(attn_mask, dtype=torch.float32),\n            \"y\": torch.tensor(s[\"target\"], dtype=torch.long),\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(\n        self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, num_layers=1, pad_idx=0\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim,\n            hidden_size=hidden,\n            batch_first=True,\n            num_layers=num_layers,\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)  # includes PAD index\n        self.pad_idx = pad_idx\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)  # [B,T,emb]\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h_last = h[-1]\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)\n        return logits\n\n\ndef collate_fn(batch):\n    keys = batch[0].keys()\n    out = {k: torch.stack([b[k] for b in batch], dim=0) for k in keys}\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys, preds_top1, preds_probs = [], [], []\n    top3_correct = 0\n    n_total = 0\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk_idx = torch.topk(probs, k=k_val, dim=1)\n            ys.extend(batch[\"y\"].detach().cpu().tolist())\n            preds_top1.extend(top1.detach().cpu().tolist())\n            preds_probs.append(probs.detach().cpu().numpy())\n            # top-3 correctness\n            for i in range(batch[\"y\"].size(0)):\n                if batch[\"y\"][i].item() in topk_idx[i].detach().cpu().tolist():\n                    top3_correct += 1\n            n_total += batch[\"y\"].size(0)\n    avg_loss = total_loss / max(1, n_total)\n    y_true = np.array(ys)\n    y_pred = np.array(preds_top1)\n    mask = y_true != pad_idx\n    y_true = y_true[mask]\n    y_pred = y_pred[mask]\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except Exception:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n_total))\n    probs_concat = (\n        np.concatenate(preds_probs, axis=0)\n        if len(preds_probs) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return avg_loss, acc, f1, top3, y_true, y_pred, probs_concat\n\n\ndef train_one_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    # Time-based split\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    # Build samples across all to get vocab; we'll re-normalize with train stats\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    # Filter per split\n    samples_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    samples_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    samples_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # Recompute normalization using train samples only\n    if len(samples_train) > 0:\n        concat_feats = [\n            s[\"seq_feats\"] for s in samples_train if s[\"seq_feats\"].shape[0] > 0\n        ]\n        if len(concat_feats) > 0:\n            all_feats = np.concatenate(concat_feats, axis=0)\n            dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n            ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n\n            def norm_samples(samples):\n                for s in samples:\n                    if s[\"seq_feats\"].shape[0] > 0:\n                        s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                        s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n            norm_samples(samples_train)\n            norm_samples(samples_val)\n            norm_samples(samples_test)\n    print(\n        f\"Samples train/val/test: {len(samples_train)}/{len(samples_val)}/{len(samples_test)}; vocab={len(act2id)}\"\n    )\n    if len(samples_train) == 0 or len(act2id) < 2:\n        print(\"Not enough data to train. Skipping.\")\n        return\n    ds_train = PrefixDataset(\n        samples_train, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_val = PrefixDataset(\n        samples_val, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_test = PrefixDataset(\n        samples_test, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    dl_train = DataLoader(\n        ds_train,\n        batch_size=batch_size,\n        shuffle=True,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_val = DataLoader(\n        ds_val,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_test = DataLoader(\n        ds_test,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n\n    # Model\n    model = LSTMBaseline(\n        vocab_size=len(act2id),\n        emb_dim=64,\n        cont_dim=5,\n        hidden=128,\n        num_layers=1,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss().to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n\n    # Training loop\n    best_val_top3 = -1.0\n    best_state = None\n    hist = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for epoch in range(1, max_epochs + 1):\n        model.train()\n        total = 0\n        running_loss = 0.0\n        for batch in dl_train:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            running_loss += loss.item() * logits.size(0)\n            total += logits.size(0)\n        train_loss = running_loss / max(1, total)\n        val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n            model, dl_val, criterion, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        hist[\"train_loss\"].append(train_loss)\n        hist[\"val_loss\"].append(val_loss)\n        hist[\"val_top3\"].append(val_top3)\n        experiment_data[name][\"losses\"][\"train\"].append((epoch, train_loss))\n        experiment_data[name][\"losses\"][\"val\"].append((epoch, val_loss))\n        experiment_data[name][\"metrics\"][\"val\"].append(\n            (epoch, {\"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3})\n        )\n        experiment_data[name][\"epochs\"].append(epoch)\n        if val_top3 > best_val_top3:\n            best_val_top3 = val_top3\n            best_state = {k: v.cpu().clone() for k, v in model.state_dict().items()}\n\n    # Load best\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n\n    # Final eval on train/val/test\n    train_loss, train_acc, train_f1, train_top3, _, _, _ = evaluate(\n        model, dl_train, criterion, device, len(act2id), pad_id\n    )\n    val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n        model, dl_val, criterion, device, len(act2id), pad_id\n    )\n    test_loss, test_acc, test_f1, test_top3, y_true_t, y_pred_t, probs_t = evaluate(\n        model, dl_test, criterion, device, len(act2id), pad_id\n    )\n    print(\n        f\"[{name}] Train: loss={train_loss:.4f} acc={train_acc:.4f} f1={train_f1:.4f} top3={train_top3:.4f}\"\n    )\n    print(\n        f\"[{name}] Test:  loss={test_loss:.4f} acc={test_acc:.4f} f1={test_f1:.4f} top3={test_top3:.4f}\"\n    )\n\n    # Save metrics\n    experiment_data[name][\"metrics\"][\"train\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": train_loss,\n                \"acc\": train_acc,\n                \"macro_f1\": train_f1,\n                \"top3\": train_top3,\n            },\n        )\n    )\n    experiment_data[name][\"metrics\"][\"val\"].append(\n        (\n            \"final\",\n            {\"loss\": val_loss, \"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3},\n        )\n    )\n    experiment_data[name][\"metrics\"][\"test\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": test_loss,\n                \"acc\": test_acc,\n                \"macro_f1\": test_f1,\n                \"top3\": test_top3,\n            },\n        )\n    )\n    experiment_data[name][\"predictions\"] = y_pred_t.tolist()\n    experiment_data[name][\"ground_truth\"] = y_true_t.tolist()\n\n    # Plots\n    try:\n        plt.figure()\n        plt.plot(hist[\"train_loss\"], label=\"train_loss\")\n        plt.plot(hist[\"val_loss\"], label=\"val_loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"loss_curves_{name}.png\"))\n        plt.close()\n\n        plt.figure()\n        plt.plot(hist[\"val_top3\"], label=\"val_top3\")\n        plt.legend()\n        plt.title(f\"Val Top-3 Acc - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Acc\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"val_top3_{name}.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n\n    # Save confusion matrix-like data (optional)\n    try:\n        from sklearn.metrics import confusion_matrix\n\n        cm = confusion_matrix(y_true_t, y_pred_t)\n        np.save(os.path.join(working_dir, f\"cm_{name}.npy\"), cm)\n    except Exception as e:\n        print(f\"[warn] Confusion matrix failed: {e}\")\n\n\ndef main():\n    datasets = load_datasets()\n    # Loop through loaded datasets; cap to 5000 earliest cases for speed\n    for key, df in datasets.items():\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep_cases = set(starts.iloc[:5000][\"case_id\"])\n                df_small = df[df[\"case_id\"].isin(keep_cases)].copy()\n            else:\n                df_small = df\n        except Exception:\n            df_small = df\n        train_one_dataset(\n            key, df_small, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    np.savez_compressed(\n        os.path.join(working_dir, \"experiment_data_compressed.npz\"),\n        data=experiment_data,\n    )\n\n\n# Execute immediately\nmain()\n","plan":"The crash arises because timestamps within each case were kept as a pandas Series and indexed with integer labels, which pandas interprets as label-based indexing; for non-RangeIndex groups, accessing ts[k] raises KeyError. I fix this by converting timestamps to a dense numpy array (ts = ... .to_numpy()), ensuring positional indexing. I also make the prefix generation loop explicitly stop at T-1 to guarantee the existence of a next event, preventing off-by-one mistakes. The rest of the pipeline remains intact: time-based split by case start, prefix features, a simple LSTM baseline, and required BPM metrics and saving.","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import (\n    f1_score,\n    accuracy_score,\n    confusion_matrix,\n    precision_recall_curve,\n    average_precision_score,\n)\nfrom sklearn.preprocessing import label_binarize\nimport random\nimport math\nfrom collections import defaultdict\n\n# Reproducibility and device\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# --------- Data loading from local XES via pm4py ----------\ndef load_xes_folder(data_dir=\"data\"):\n    datasets = {}\n    try:\n        import pm4py\n    except Exception as e:\n        print(f\"pm4py not available: {e}\")\n        return datasets\n    if not os.path.isdir(data_dir):\n        print(f\"Data directory not found: {data_dir}\")\n        return datasets\n    for fn in os.listdir(data_dir):\n        if fn.lower().endswith(\".xes\") or fn.lower().endswith(\".xes.gz\"):\n            path = os.path.join(data_dir, fn)\n            try:\n                log = pm4py.read_xes(path)\n                df = pm4py.convert_to_dataframe(log)\n                # Standardize columns\n                # pm4py dataframe typically has case:concept:name, concept:name, time:timestamp, lifecycle:transition\n                cols = df.columns\n                case_col = (\n                    \"case:concept:name\"\n                    if \"case:concept:name\" in cols\n                    else (\"case\" if \"case\" in cols else None)\n                )\n                act_col = (\n                    \"concept:name\"\n                    if \"concept:name\" in cols\n                    else (\"activity\" if \"activity\" in cols else None)\n                )\n                ts_col = (\n                    \"time:timestamp\"\n                    if \"time:timestamp\" in cols\n                    else (\"timestamp\" if \"timestamp\" in cols else None)\n                )\n                life_col = (\n                    \"lifecycle:transition\"\n                    if \"lifecycle:transition\" in cols\n                    else (\"lifecycle\" if \"lifecycle\" in cols else None)\n                )\n                if case_col is None or act_col is None or ts_col is None:\n                    print(f\"Missing required columns in {fn}, skipping.\")\n                    continue\n                out = pd.DataFrame(\n                    {\n                        \"case_id\": df[case_col].astype(str).values,\n                        \"activity\": df[act_col].astype(str).values,\n                        \"timestamp\": pd.to_datetime(df[ts_col], utc=True),\n                    }\n                )\n                if life_col is not None:\n                    out[\"lifecycle\"] = df[life_col].astype(str).values\n                name = os.path.splitext(fn)[0]\n                datasets[name] = out\n                print(\n                    f\"Loaded {name}: {len(out)} events, {out['case_id'].nunique()} cases\"\n                )\n            except Exception as e:\n                print(f\"Failed to load {fn}: {e}\")\n    return datasets\n\n\n# --------- Prefix building and split ----------\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        mask = df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")\n        if mask.any():\n            df = df[mask]\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        if len(g) < 2:\n            continue\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        ts = (g_ts.astype(\"int64\") // 10**9).to_numpy(np.int64)\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str)], dtype=np.int64\n        )\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(np.float32)\n        T = len(acts_ids)\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": acts_ids[:k].tolist(),\n                    \"seq_feats\": feats[:k].copy(),\n                    \"target\": int(acts_ids[k]),\n                    \"prefix_len\": k,\n                }\n            )\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    for s in samples:\n        pass  # initial no norm; will norm on train split\n    return samples, act2id, id2act, pad_id\n\n\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.vstack(\n            [\n                np.zeros((pad_len, self.num_cont), dtype=np.float32),\n                feats.astype(np.float32),\n            ]\n        )\n        attn = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad).long(),\n            \"feats\": torch.tensor(feats_pad).float(),\n            \"mask\": torch.tensor(attn).float(),\n            \"y\": torch.tensor(s[\"target\"]).long(),\n            \"prefix_len\": L,\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, pad_idx=0):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim, hidden_size=hidden, batch_first=True\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h = self.dropout(h[-1])\n        return self.fc(h)\n\n\ndef collate_fn(batch):\n    out = {\n        k: (\n            torch.stack([b[k] for b in batch], 0)\n            if isinstance(batch[0][k], torch.Tensor)\n            else [b[k] for b in batch]\n        )\n        for k in batch[0].keys()\n    }\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys = []\n    yhat = []\n    probs_list = []\n    n = 0\n    top3_correct = 0\n    pref_lens = []\n    top3_flags = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk = torch.topk(probs, k=k_val, dim=1)\n            y = batch[\"y\"]\n            ys.extend(y.detach().cpu().tolist())\n            yhat.extend(top1.detach().cpu().tolist())\n            probs_list.append(probs.detach().cpu().numpy())\n            for i in range(y.size(0)):\n                flag = int(y[i].item() in topk[i].detach().cpu().tolist())\n                top3_correct += flag\n                top3_flags.append(flag)\n                pref_lens.append(int(batch[\"prefix_len\"][i].item()))\n            n += y.size(0)\n    avg_loss = total_loss / max(1, n)\n    y_true = np.array(ys)\n    y_pred = np.array(yhat)\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n))\n    probs_concat = (\n        np.concatenate(probs_list, axis=0)\n        if len(probs_list) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return (\n        avg_loss,\n        acc,\n        f1,\n        top3,\n        y_true,\n        y_pred,\n        probs_concat,\n        np.array(pref_lens),\n        np.array(top3_flags),\n    )\n\n\ndef train_on_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    s_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    s_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    s_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # normalize time features on train\n    if len(s_train) > 0:\n        feats = np.concatenate(\n            [s[\"seq_feats\"] for s in s_train if len(s[\"seq_feats\"]) > 0], axis=0\n        )\n        dt_mean, dt_std = feats[:, 0].mean(), feats[:, 0].std() + 1e-6\n        ss_mean, ss_std = feats[:, 1].mean(), feats[:, 1].std() + 1e-6\n\n        def norm(samples):\n            for s in samples:\n                if s[\"seq_feats\"].shape[0] > 0:\n                    s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                    s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n        norm(s_train)\n        norm(s_val)\n        norm(s_test)\n    print(\n        f\"Samples train/val/test: {len(s_train)}/{len(s_val)}/{len(s_test)}; vocab={len(act2id)}\"\n    )\n    if len(s_train) == 0 or len(act2id) < 2:\n        print(\"Insufficient data; skipping.\")\n        return None\n    ds_tr = PrefixDataset(s_train, pad_id, max_prefix_len, 5)\n    ds_va = PrefixDataset(s_val, pad_id, max_prefix_len, 5)\n    ds_te = PrefixDataset(s_test, pad_id, max_prefix_len, 5)\n    dl_tr = DataLoader(\n        ds_tr, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n    )\n    dl_va = DataLoader(\n        ds_va, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    dl_te = DataLoader(\n        ds_te, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    model = LSTMBaseline(\n        vocab_size=len(act2id), emb_dim=64, cont_dim=5, hidden=128, pad_idx=pad_id\n    ).to(device)\n    crit = nn.CrossEntropyLoss().to(device)\n    opt = torch.optim.Adam(model.parameters(), lr=lr)\n    best_top3 = -1.0\n    best_state = None\n    history = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for ep in range(1, max_epochs + 1):\n        model.train()\n        tot = 0\n        run_loss = 0.0\n        for batch in dl_tr:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            opt.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = crit(logits, batch[\"y\"])\n            loss.backward()\n            opt.step()\n            run_loss += loss.item() * logits.size(0)\n            tot += logits.size(0)\n        tr_loss = run_loss / max(1, tot)\n        va_loss, va_acc, va_f1, va_top3, *_ = evaluate(\n            model, dl_va, crit, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {ep}: val_loss={va_loss:.4f} acc={va_acc:.4f} f1={va_f1:.4f} top3={va_top3:.4f}\"\n        )\n        history[\"train_loss\"].append(tr_loss)\n        history[\"val_loss\"].append(va_loss)\n        history[\"val_top3\"].append(va_top3)\n        if va_top3 > best_top3:\n            best_top3 = va_top3\n            best_state = {\n                k: v.detach().cpu().clone() for k, v in model.state_dict().items()\n            }\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n    tr_loss, tr_acc, tr_f1, tr_top3, *_ = evaluate(\n        model, dl_tr, crit, device, len(act2id), pad_id\n    )\n    te_loss, te_acc, te_f1, te_top3, y_true, y_pred, probs, pref_lens, top3_flags = (\n        evaluate(model, dl_te, crit, device, len(act2id), pad_id)\n    )\n    print(\n        f\"[{name}] Test: loss={te_loss:.4f} acc={te_acc:.4f} f1={te_f1:.4f} top3={te_top3:.4f}\"\n    )\n    # package experiment data\n    exp = {\n        \"metrics\": {\n            \"train\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": tr_loss,\n                        \"acc\": tr_acc,\n                        \"macro_f1\": tr_f1,\n                        \"top3\": tr_top3,\n                    },\n                )\n            ],\n            \"val\": [],\n            \"test\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": te_loss,\n                        \"acc\": te_acc,\n                        \"macro_f1\": te_f1,\n                        \"top3\": te_top3,\n                    },\n                )\n            ],\n        },\n        \"losses\": {\n            \"train\": list(enumerate(history[\"train_loss\"], start=1)),\n            \"val\": list(enumerate(history[\"val_loss\"], start=1)),\n        },\n        \"predictions\": y_pred.tolist(),\n        \"ground_truth\": y_true.tolist(),\n        \"epochs\": list(range(1, len(history[\"train_loss\"]) + 1)),\n        \"probs\": probs,\n        \"prefix_lens\": pref_lens.tolist(),\n        \"top3_flags\": top3_flags.tolist(),\n        \"act2id\": act2id,\n    }\n    # plots for this dataset\n    try:\n        plt.figure()\n        plt.plot(history[\"train_loss\"], label=\"train\")\n        plt.plot(history[\"val_loss\"], label=\"val\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating loss curves for {name}: {e}\")\n        plt.close()\n    try:\n        cm = confusion_matrix(y_true, y_pred)\n        plt.figure(figsize=(5, 4))\n        plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n        plt.colorbar()\n        plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n        plt.xlabel(\"Predicted\")\n        plt.ylabel(\"True\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_confusion_matrix.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating confusion matrix for {name}: {e}\")\n        plt.close()\n    try:\n        # Top-3 accuracy vs prefix length\n        if len(pref_lens) > 0:\n            d = defaultdict(list)\n            for L, flag in zip(pref_lens, top3_flags):\n                d[int(L)].append(int(flag))\n            xs = sorted(d.keys())\n            ys = [np.mean(d[k]) for k in xs]\n            plt.figure()\n            plt.plot(xs, ys, marker=\"o\")\n            plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n            plt.xlabel(\"Prefix Length\")\n            plt.ylabel(\"Top-3 Accuracy\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_top3_vs_prefixlen.png\"))\n            plt.close()\n    except Exception as e:\n        print(f\"Error creating Top-3 vs prefix length for {name}: {e}\")\n        plt.close()\n    try:\n        # Macro PR curve (one-vs-rest); may be coarse due to many classes\n        if probs.shape[0] > 0:\n            classes = np.unique(y_true)\n            Y = label_binarize(y_true, classes=range(probs.shape[1]))\n            # only keep columns present in classes to avoid PAD\n            present = [c for c in classes]\n            if len(present) > 1:\n                precisions = []\n                recalls = []\n                aps = []\n                for c in present:\n                    p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                    ap = average_precision_score(Y[:, c], probs[:, c])\n                    precisions.append(\n                        np.interp(np.linspace(0, 1, 101), r[::-1], p[::-1])\n                    )\n                    recalls.append(np.linspace(0, 1, 101))\n                    aps.append(ap)\n                macro_p = np.mean(np.stack(precisions, 0), 0)\n                macro_r = np.linspace(0, 1, 101)\n                plt.figure()\n                plt.plot(macro_r, macro_p, label=f\"Macro-PR (mAP={np.mean(aps):.3f})\")\n                plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Recall\")\n                plt.ylabel(\"Precision\")\n                plt.legend()\n                plt.tight_layout()\n                plt.savefig(os.path.join(working_dir, f\"{name}_macro_pr.png\"))\n                plt.close()\n    except Exception as e:\n        print(f\"Error creating PR curve for {name}: {e}\")\n        plt.close()\n    return name, exp\n\n\ndef main():\n    datasets = load_xes_folder(data_dir=os.path.join(os.getcwd(), \"data\"))\n    experiment_data = {}\n    for name, df in datasets.items():\n        # optional cap earliest 5000 cases\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep = set(starts.iloc[:5000][\"case_id\"])\n                df = df[df[\"case_id\"].isin(keep)].copy()\n        except:\n            pass\n        res = train_on_dataset(\n            name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n        if res is not None:\n            k, exp = res\n            experiment_data[k] = exp\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Print evaluation metrics\n    for k, v in experiment_data.items():\n        test_metrics = dict(v[\"metrics\"][\"test\"][0][1])\n        print(\n            f\"{k} | Test acc={test_metrics['acc']:.4f} macro_f1={test_metrics['macro_f1']:.4f} top3={test_metrics['top3']:.4f} loss={test_metrics['loss']:.4f}\"\n        )\n\n    # Secondary plotting pass strictly from experiment_data.npy (as required)\n    try:\n        experiment_data_loaded = np.load(\n            os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n        ).item()\n    except Exception as e:\n        print(f\"Error loading experiment data: {e}\")\n        experiment_data_loaded = {}\n    for name, ed in experiment_data_loaded.items():\n        try:\n            # re-plot loss curves from saved data\n            plt.figure()\n            tl = [y for (_, y) in ed.get(\"losses\", {}).get(\"train\", [])]\n            vl = [y for (_, y) in ed.get(\"losses\", {}).get(\"val\", [])]\n            if len(tl) > 0:\n                plt.plot(tl, label=\"train\")\n            if len(vl) > 0:\n                plt.plot(vl, label=\"val\")\n            plt.legend()\n            plt.title(f\"Loss Curves - {name}\\nNext-activity\")\n            plt.xlabel(\"Epoch\")\n            plt.ylabel(\"Loss\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves_reload.png\"))\n            plt.close()\n        except Exception as e:\n            print(f\"Error creating plot1: {e}\")\n            plt.close()\n        try:\n            # confusion matrix from predictions and ground truth\n            y_true = ed.get(\"ground_truth\", [])\n            y_pred = ed.get(\"predictions\", [])\n            if len(y_true) > 0 and len(y_pred) > 0:\n                cm = confusion_matrix(y_true, y_pred)\n                plt.figure(figsize=(5, 4))\n                plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n                plt.colorbar()\n                plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Predicted\")\n                plt.ylabel(\"True\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_confusion_matrix_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot2: {e}\")\n            plt.close()\n        try:\n            # Top-3 vs prefix length if present\n            pref = ed.get(\"prefix_lens\", [])\n            flags = ed.get(\"top3_flags\", [])\n            if len(pref) > 0 and len(flags) > 0:\n                d = defaultdict(list)\n                for L, f in zip(pref, flags):\n                    d[int(L)].append(int(f))\n                xs = sorted(d.keys())\n                ys = [float(np.mean(d[x])) for x in xs]\n                plt.figure()\n                plt.plot(xs, ys, marker=\"o\")\n                plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n                plt.xlabel(\"Prefix Length\")\n                plt.ylabel(\"Top-3 Accuracy\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_top3_vs_prefixlen_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot3: {e}\")\n            plt.close()\n        try:\n            # Macro PR curve if probs available\n            probs = np.array(ed.get(\"probs\", []))\n            y_true = ed.get(\"ground_truth\", [])\n            if probs.size > 0 and len(y_true) > 0:\n                classes = sorted(set(y_true))\n                Y = label_binarize(np.array(y_true), classes=range(probs.shape[1]))\n                present = classes\n                if len(present) > 1:\n                    precisions = []\n                    aps = []\n                    grid = np.linspace(0, 1, 101)\n                    for c in present:\n                        p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                        precisions.append(np.interp(grid, r[::-1], p[::-1]))\n                        aps.append(average_precision_score(Y[:, c], probs[:, c]))\n                    macro_p = np.mean(np.stack(precisions, 0), 0)\n                    plt.figure()\n                    plt.plot(grid, macro_p, label=f\"mAP={np.mean(aps):.3f}\")\n                    plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                    plt.xlabel(\"Recall\")\n                    plt.ylabel(\"Precision\")\n                    plt.legend()\n                    plt.tight_layout()\n                    plt.savefig(\n                        os.path.join(working_dir, f\"{name}_macro_pr_reload.png\")\n                    )\n                    plt.close()\n        except Exception as e:\n            print(f\"Error creating plot4: {e}\")\n            plt.close()\n\n\nif __name__ == \"__main__\":\n    main()\n","plot_plan":"We tackle next-activity prediction from local XES logs using pm4py, building time-ordered prefixes and splitting cases by start time into train/val/test. Activities are tokenized and simple temporal/calendar features are added; a small LSTM consumes past activities plus features to predict the next activity. We train for a few epochs, select the best model by validation Top-3 accuracy, then evaluate on test with accuracy, macro-F1, Top-3, and confusion matrix; we also keep per-sample prefix lengths to compute Top-3 vs prefix length. We aggregate results into an experiment_data object and save it. Finally, we generate standard plots only from the saved experiment_data: loss curves, confusion matrix, macro-PR if probabilities exist, and Top-3 vs prefix length. We ensure each plot is guarded with try-except and saved to working_dir, closing figures after saving. The code uses pm4py to load any local .xes files, avoids external datasets, performs time-based splits, and reports BPM metrics.","step":4,"id":"726b2721d45c4800b9381c5d265fefcb","ctime":1757756714.9796784,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available in /workspace/data: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<22:13,  9.81it/s]","\rparsing log, completed traces ::   1%|1         | 155/13087 [00:00<00:14, 902.77it/s]","\rparsing log, completed traces ::   2%|2         | 307/13087 [00:00<00:10, 1183.23it/s]","\rparsing log, completed traces ::   4%|3         | 475/13087 [00:00<00:09, 1374.93it/s]","\rparsing log, completed traces ::   5%|4         | 632/13087 [00:00<00:08, 1441.90it/s]","\rparsing log, completed traces ::   6%|5         | 777/13087 [00:00<00:15, 819.10it/s] ","\rparsing log, completed traces ::   7%|7         | 927/13087 [00:00<00:12, 964.33it/s]","\rparsing log, completed traces ::   8%|8         | 1092/13087 [00:01<00:10, 1124.33it/s]","\rparsing log, completed traces ::  10%|9         | 1260/13087 [00:01<00:09, 1261.95it/s]","\rparsing log, completed traces ::  11%|#1        | 1456/13087 [00:01<00:08, 1445.90it/s]","\rparsing log, completed traces ::  12%|#2        | 1635/13087 [00:01<00:07, 1535.07it/s]","\rparsing log, completed traces ::  14%|#3        | 1809/13087 [00:01<00:07, 1587.27it/s]","\rparsing log, completed traces ::  15%|#5        | 1977/13087 [00:01<00:07, 1564.32it/s]","\rparsing log, completed traces ::  16%|#6        | 2142/13087 [00:01<00:06, 1586.66it/s]","\rparsing log, completed traces ::  18%|#7        | 2311/13087 [00:01<00:06, 1614.55it/s]","\rparsing log, completed traces ::  19%|#9        | 2499/13087 [00:01<00:06, 1691.18it/s]","\rparsing log, completed traces ::  20%|##        | 2671/13087 [00:01<00:06, 1682.60it/s]","\rparsing log, completed traces ::  22%|##1       | 2842/13087 [00:02<00:06, 1609.51it/s]","\rparsing log, completed traces ::  23%|##2       | 3005/13087 [00:02<00:10, 939.23it/s] ","\rparsing log, completed traces ::  24%|##4       | 3153/13087 [00:02<00:09, 1042.68it/s]","\rparsing log, completed traces ::  26%|##5       | 3396/13087 [00:02<00:07, 1339.99it/s]","\rparsing log, completed traces ::  27%|##7       | 3575/13087 [00:02<00:06, 1442.75it/s]","\rparsing log, completed traces ::  29%|##8       | 3764/13087 [00:02<00:05, 1554.54it/s]","\rparsing log, completed traces ::  30%|###       | 3945/13087 [00:02<00:05, 1621.59it/s]","\rparsing log, completed traces ::  32%|###1      | 4186/13087 [00:03<00:04, 1836.31it/s]","\rparsing log, completed traces ::  34%|###3      | 4403/13087 [00:03<00:04, 1924.51it/s]","\rparsing log, completed traces ::  35%|###5      | 4608/13087 [00:03<00:04, 1960.03it/s]","\rparsing log, completed traces ::  37%|###6      | 4811/13087 [00:03<00:04, 1910.52it/s]","\rparsing log, completed traces ::  38%|###8      | 5012/13087 [00:03<00:04, 1938.09it/s]","\rparsing log, completed traces ::  40%|###9      | 5210/13087 [00:03<00:04, 1865.51it/s]","\rparsing log, completed traces ::  41%|####1     | 5400/13087 [00:03<00:04, 1822.65it/s]","\rparsing log, completed traces ::  43%|####2     | 5585/13087 [00:03<00:04, 1711.36it/s]","\rparsing log, completed traces ::  44%|####4     | 5764/13087 [00:03<00:04, 1726.93it/s]","\rparsing log, completed traces ::  45%|####5     | 5939/13087 [00:04<00:07, 943.97it/s] ","\rparsing log, completed traces ::  47%|####6     | 6102/13087 [00:04<00:06, 1066.47it/s]","\rparsing log, completed traces ::  48%|####8     | 6284/13087 [00:04<00:05, 1219.00it/s]","\rparsing log, completed traces ::  50%|####9     | 6481/13087 [00:04<00:04, 1381.28it/s]","\rparsing log, completed traces ::  51%|#####     | 6648/13087 [00:04<00:04, 1440.78it/s]","\rparsing log, completed traces ::  52%|#####2    | 6813/13087 [00:04<00:04, 1491.87it/s]","\rparsing log, completed traces ::  53%|#####3    | 6978/13087 [00:04<00:04, 1478.08it/s]","\rparsing log, completed traces ::  55%|#####4    | 7150/13087 [00:05<00:03, 1542.98it/s]","\rparsing log, completed traces ::  56%|#####5    | 7313/13087 [00:05<00:03, 1554.25it/s]","\rparsing log, completed traces ::  57%|#####7    | 7475/13087 [00:05<00:03, 1511.56it/s]","\rparsing log, completed traces ::  58%|#####8    | 7631/13087 [00:05<00:03, 1506.49it/s]","\rparsing log, completed traces ::  60%|#####9    | 7795/13087 [00:05<00:03, 1542.51it/s]","\rparsing log, completed traces ::  61%|######    | 7952/13087 [00:05<00:03, 1522.31it/s]","\rparsing log, completed traces ::  62%|######2   | 8125/13087 [00:05<00:03, 1574.39it/s]","\rparsing log, completed traces ::  63%|######3   | 8305/13087 [00:05<00:02, 1638.91it/s]","\rparsing log, completed traces ::  65%|######4   | 8476/13087 [00:05<00:02, 1659.72it/s]","\rparsing log, completed traces ::  66%|######6   | 8645/13087 [00:05<00:02, 1666.27it/s]","\rparsing log, completed traces ::  67%|######7   | 8813/13087 [00:06<00:02, 1653.27it/s]","\rparsing log, completed traces ::  69%|######8   | 8979/13087 [00:06<00:04, 844.36it/s] ","\rparsing log, completed traces ::  70%|######9   | 9120/13087 [00:06<00:04, 943.26it/s]","\rparsing log, completed traces ::  71%|#######1  | 9308/13087 [00:06<00:03, 1131.69it/s]","\rparsing log, completed traces ::  72%|#######2  | 9471/13087 [00:06<00:02, 1241.38it/s]","\rparsing log, completed traces ::  74%|#######3  | 9624/13087 [00:06<00:02, 1280.12it/s]","\rparsing log, completed traces ::  75%|#######4  | 9787/13087 [00:06<00:02, 1365.32it/s]","\rparsing log, completed traces ::  76%|#######6  | 9950/13087 [00:07<00:02, 1433.72it/s]","\rparsing log, completed traces ::  77%|#######7  | 10112/13087 [00:07<00:02, 1482.43it/s]","\rparsing log, completed traces ::  78%|#######8  | 10270/13087 [00:07<00:01, 1492.40it/s]","\rparsing log, completed traces ::  80%|#######9  | 10426/13087 [00:07<00:01, 1495.09it/s]","\rparsing log, completed traces ::  81%|########  | 10580/13087 [00:07<00:01, 1456.05it/s]","\rparsing log, completed traces ::  82%|########2 | 10761/13087 [00:07<00:01, 1552.73it/s]","\rparsing log, completed traces ::  84%|########3 | 10946/13087 [00:07<00:01, 1636.64it/s]","\rparsing log, completed traces ::  85%|########4 | 11121/13087 [00:07<00:01, 1667.60it/s]","\rparsing log, completed traces ::  86%|########6 | 11295/13087 [00:07<00:01, 1683.98it/s]","\rparsing log, completed traces ::  88%|########7 | 11497/13087 [00:08<00:00, 1780.59it/s]","\rparsing log, completed traces ::  89%|########9 | 11677/13087 [00:08<00:00, 1764.28it/s]","\rparsing log, completed traces ::  91%|######### | 11879/13087 [00:08<00:00, 1839.23it/s]","\rparsing log, completed traces ::  92%|#########2| 12064/13087 [00:08<00:00, 1838.87it/s]","\rparsing log, completed traces ::  94%|#########3| 12278/13087 [00:08<00:00, 1928.19it/s]","\rparsing log, completed traces ::  95%|#########5| 12472/13087 [00:08<00:00, 879.69it/s] ","\rparsing log, completed traces ::  97%|#########6| 12671/13087 [00:09<00:00, 1058.76it/s]","\rparsing log, completed traces ::  98%|#########8| 12863/13087 [00:09<00:00, 1218.85it/s]","\rparsing log, completed traces :: 100%|#########9| 13065/13087 [00:09<00:00, 1387.49it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:09<00:00, 1419.98it/s]","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<4:44:28,  1.85it/s]","\rparsing log, completed traces ::   0%|          | 68/31509 [00:00<03:43, 140.65it/s]","\rparsing log, completed traces ::   0%|          | 126/31509 [00:00<02:10, 240.46it/s]","\rparsing log, completed traces ::   1%|          | 194/31509 [00:00<01:29, 348.13it/s]","\rparsing log, completed traces ::   1%|          | 254/31509 [00:00<01:15, 412.78it/s]","\rparsing log, completed traces ::   1%|1         | 318/31509 [00:01<01:06, 472.43it/s]","\rparsing log, completed traces ::   1%|1         | 382/31509 [00:01<01:00, 518.19it/s]","\rparsing log, completed traces ::   1%|1         | 443/31509 [00:01<00:57, 538.22it/s]","\rparsing log, completed traces ::   2%|1         | 507/31509 [00:01<00:54, 566.63it/s]","\rparsing log, completed traces ::   2%|1         | 569/31509 [00:01<00:54, 565.64it/s]","\rparsing log, completed traces ::   2%|2         | 635/31509 [00:01<00:52, 591.58it/s]","\rparsing log, completed traces ::   2%|2         | 707/31509 [00:01<00:49, 626.99it/s]","\rparsing log, completed traces ::   2%|2         | 772/31509 [00:01<00:48, 628.07it/s]","\rparsing log, completed traces ::   3%|2         | 837/31509 [00:01<00:49, 615.97it/s]","\rparsing log, completed traces ::   3%|2         | 901/31509 [00:01<00:49, 619.32it/s]","\rparsing log, completed traces ::   3%|3         | 973/31509 [00:02<00:47, 646.04it/s]","\rparsing log, completed traces ::   3%|3         | 1039/31509 [00:02<00:46, 648.56it/s]","\rparsing log, completed traces ::   4%|3         | 1105/31509 [00:02<00:47, 635.62it/s]","\rparsing log, completed traces ::   4%|3         | 1172/31509 [00:02<00:47, 644.40it/s]","\rparsing log, completed traces ::   4%|3         | 1241/31509 [00:02<00:46, 654.84it/s]","\rparsing log, completed traces ::   4%|4         | 1307/31509 [00:02<00:46, 649.07it/s]","\rparsing log, completed traces ::   4%|4         | 1374/31509 [00:02<00:46, 653.26it/s]","\rparsing log, completed traces ::   5%|4         | 1440/31509 [00:03<01:27, 343.66it/s]","\rparsing log, completed traces ::   5%|4         | 1502/31509 [00:03<01:16, 392.03it/s]","\rparsing log, completed traces ::   5%|4         | 1568/31509 [00:03<01:07, 446.03it/s]","\rparsing log, completed traces ::   5%|5         | 1633/31509 [00:03<01:00, 490.91it/s]","\rparsing log, completed traces ::   5%|5         | 1694/31509 [00:03<00:57, 517.11it/s]","\rparsing log, completed traces ::   6%|5         | 1759/31509 [00:03<00:53, 551.06it/s]","\rparsing log, completed traces ::   6%|5         | 1821/31509 [00:03<00:53, 556.22it/s]","\rparsing log, completed traces ::   6%|5         | 1881/31509 [00:03<00:52, 565.73it/s]","\rparsing log, completed traces ::   6%|6         | 1946/31509 [00:03<00:50, 587.82it/s]","\rparsing log, completed traces ::   6%|6         | 2011/31509 [00:04<00:48, 603.34it/s]","\rparsing log, completed traces ::   7%|6         | 2074/31509 [00:04<00:48, 609.16it/s]","\rparsing log, completed traces ::   7%|6         | 2138/31509 [00:04<00:47, 617.99it/s]","\rparsing log, completed traces ::   7%|6         | 2201/31509 [00:04<00:47, 621.42it/s]","\rparsing log, completed traces ::   7%|7         | 2272/31509 [00:04<00:45, 645.06it/s]","\rparsing log, completed traces ::   7%|7         | 2338/31509 [00:04<00:45, 647.48it/s]","\rparsing log, completed traces ::   8%|7         | 2404/31509 [00:04<00:45, 644.97it/s]","\rparsing log, completed traces ::   8%|7         | 2471/31509 [00:04<00:44, 646.03it/s]","\rparsing log, completed traces ::   8%|8         | 2536/31509 [00:05<01:22, 351.71it/s]","\rparsing log, completed traces ::   8%|8         | 2610/31509 [00:05<01:08, 423.18it/s]","\rparsing log, completed traces ::   8%|8         | 2677/31509 [00:05<01:00, 473.98it/s]","\rparsing log, completed traces ::   9%|8         | 2739/31509 [00:05<00:56, 505.61it/s]","\rparsing log, completed traces ::   9%|8         | 2806/31509 [00:05<00:52, 544.14it/s]","\rparsing log, completed traces ::   9%|9         | 2880/31509 [00:05<00:48, 593.37it/s]","\rparsing log, completed traces ::   9%|9         | 2952/31509 [00:05<00:45, 625.19it/s]","\rparsing log, completed traces ::  10%|9         | 3022/31509 [00:05<00:44, 642.80it/s]","\rparsing log, completed traces ::  10%|9         | 3090/31509 [00:05<00:44, 644.48it/s]","\rparsing log, completed traces ::  10%|#         | 3157/31509 [00:06<00:44, 644.26it/s]","\rparsing log, completed traces ::  10%|#         | 3224/31509 [00:06<00:44, 634.95it/s]","\rparsing log, completed traces ::  10%|#         | 3298/31509 [00:06<00:42, 664.01it/s]","\rparsing log, completed traces ::  11%|#         | 3369/31509 [00:06<00:41, 676.01it/s]","\rparsing log, completed traces ::  11%|#         | 3446/31509 [00:06<00:40, 701.36it/s]","\rparsing log, completed traces ::  11%|#1        | 3523/31509 [00:06<00:39, 717.21it/s]","\rparsing log, completed traces ::  11%|#1        | 3596/31509 [00:06<00:40, 689.81it/s]","\rparsing log, completed traces ::  12%|#1        | 3667/31509 [00:06<00:40, 694.02it/s]","\rparsing log, completed traces ::  12%|#1        | 3737/31509 [00:06<00:39, 694.85it/s]","\rparsing log, completed traces ::  12%|#2        | 3807/31509 [00:07<01:20, 342.43it/s]","\rparsing log, completed traces ::  12%|#2        | 3873/31509 [00:07<01:09, 396.29it/s]","\rparsing log, completed traces ::  13%|#2        | 3943/31509 [00:07<01:00, 454.62it/s]","\rparsing log, completed traces ::  13%|#2        | 4016/31509 [00:07<00:53, 512.71it/s]","\rparsing log, completed traces ::  13%|#2        | 4083/31509 [00:07<00:50, 547.78it/s]","\rparsing log, completed traces ::  13%|#3        | 4153/31509 [00:07<00:46, 584.94it/s]","\rparsing log, completed traces ::  13%|#3        | 4220/31509 [00:07<00:45, 595.66it/s]","\rparsing log, completed traces ::  14%|#3        | 4285/31509 [00:08<00:45, 595.64it/s]","\rparsing log, completed traces ::  14%|#3        | 4351/31509 [00:08<00:44, 612.07it/s]","\rparsing log, completed traces ::  14%|#4        | 4419/31509 [00:08<00:43, 629.68it/s]","\rparsing log, completed traces ::  14%|#4        | 4485/31509 [00:08<00:43, 623.50it/s]","\rparsing log, completed traces ::  14%|#4        | 4552/31509 [00:08<00:42, 636.34it/s]","\rparsing log, completed traces ::  15%|#4        | 4618/31509 [00:08<00:41, 642.87it/s]","\rparsing log, completed traces ::  15%|#4        | 4687/31509 [00:08<00:41, 652.78it/s]","\rparsing log, completed traces ::  15%|#5        | 4760/31509 [00:08<00:39, 674.45it/s]","\rparsing log, completed traces ::  15%|#5        | 4828/31509 [00:08<00:40, 665.74it/s]","\rparsing log, completed traces ::  16%|#5        | 4895/31509 [00:08<00:40, 660.34it/s]","\rparsing log, completed traces ::  16%|#5        | 4962/31509 [00:09<00:40, 656.79it/s]","\rparsing log, completed traces ::  16%|#5        | 5031/31509 [00:09<00:39, 662.89it/s]","\rparsing log, completed traces ::  16%|#6        | 5103/31509 [00:09<00:38, 679.39it/s]","\rparsing log, completed traces ::  16%|#6        | 5172/31509 [00:09<00:39, 668.19it/s]","\rparsing log, completed traces ::  17%|#6        | 5239/31509 [00:09<01:20, 326.39it/s]","\rparsing log, completed traces ::  17%|#6        | 5307/31509 [00:09<01:08, 385.03it/s]","\rparsing log, completed traces ::  17%|#7        | 5374/31509 [00:10<00:59, 439.73it/s]","\rparsing log, completed traces ::  17%|#7        | 5437/31509 [00:10<00:54, 480.66it/s]","\rparsing log, completed traces ::  17%|#7        | 5510/31509 [00:10<00:48, 539.52it/s]","\rparsing log, completed traces ::  18%|#7        | 5583/31509 [00:10<00:44, 585.18it/s]","\rparsing log, completed traces ::  18%|#7        | 5657/31509 [00:10<00:41, 625.46it/s]","\rparsing log, completed traces ::  18%|#8        | 5735/31509 [00:10<00:38, 667.34it/s]","\rparsing log, completed traces ::  18%|#8        | 5807/31509 [00:10<00:38, 674.48it/s]","\rparsing log, completed traces ::  19%|#8        | 5880/31509 [00:10<00:37, 690.19it/s]","\rparsing log, completed traces ::  19%|#8        | 5956/31509 [00:10<00:36, 707.26it/s]","\rparsing log, completed traces ::  19%|#9        | 6029/31509 [00:10<00:36, 695.11it/s]","\rparsing log, completed traces ::  19%|#9        | 6100/31509 [00:11<00:36, 687.86it/s]","\rparsing log, completed traces ::  20%|#9        | 6172/31509 [00:11<00:36, 696.73it/s]","\rparsing log, completed traces ::  20%|#9        | 6243/31509 [00:11<00:36, 699.72it/s]","\rparsing log, completed traces ::  20%|##        | 6314/31509 [00:11<00:36, 683.07it/s]","\rparsing log, completed traces ::  20%|##        | 6387/31509 [00:11<00:36, 695.49it/s]","\rparsing log, completed traces ::  20%|##        | 6457/31509 [00:11<00:36, 679.90it/s]","\rparsing log, completed traces ::  21%|##        | 6526/31509 [00:11<00:37, 666.39it/s]","\rparsing log, completed traces ::  21%|##        | 6593/31509 [00:11<00:38, 641.77it/s]","\rparsing log, completed traces ::  21%|##1       | 6660/31509 [00:11<00:38, 648.40it/s]","\rparsing log, completed traces ::  21%|##1       | 6730/31509 [00:12<00:37, 662.95it/s]","\rparsing log, completed traces ::  22%|##1       | 6797/31509 [00:12<00:37, 660.98it/s]","\rparsing log, completed traces ::  22%|##1       | 6864/31509 [00:12<01:24, 292.88it/s]","\rparsing log, completed traces ::  22%|##2       | 6935/31509 [00:12<01:08, 357.45it/s]","\rparsing log, completed traces ::  22%|##2       | 6999/31509 [00:12<01:00, 407.63it/s]","\rparsing log, completed traces ::  22%|##2       | 7074/31509 [00:12<00:51, 477.47it/s]","\rparsing log, completed traces ::  23%|##2       | 7142/31509 [00:13<00:46, 521.57it/s]","\rparsing log, completed traces ::  23%|##2       | 7207/31509 [00:13<00:44, 540.64it/s]","\rparsing log, completed traces ::  23%|##3       | 7275/31509 [00:13<00:42, 573.61it/s]","\rparsing log, completed traces ::  23%|##3       | 7340/31509 [00:13<00:41, 588.47it/s]","\rparsing log, completed traces ::  24%|##3       | 7407/31509 [00:13<00:39, 606.30it/s]","\rparsing log, completed traces ::  24%|##3       | 7478/31509 [00:13<00:37, 634.27it/s]","\rparsing log, completed traces ::  24%|##3       | 7551/31509 [00:13<00:36, 660.17it/s]","\rparsing log, completed traces ::  24%|##4       | 7620/31509 [00:13<00:36, 660.65it/s]","\rparsing log, completed traces ::  24%|##4       | 7688/31509 [00:13<00:36, 656.39it/s]","\rparsing log, completed traces ::  25%|##4       | 7755/31509 [00:13<00:35, 660.07it/s]","\rparsing log, completed traces ::  25%|##4       | 7822/31509 [00:14<00:36, 653.49it/s]","\rparsing log, completed traces ::  25%|##5       | 7894/31509 [00:14<00:35, 671.13it/s]","\rparsing log, completed traces ::  25%|##5       | 7964/31509 [00:14<00:34, 675.45it/s]","\rparsing log, completed traces ::  26%|##5       | 8035/31509 [00:14<00:34, 682.61it/s]","\rparsing log, completed traces ::  26%|##5       | 8104/31509 [00:14<00:34, 684.68it/s]","\rparsing log, completed traces ::  26%|##5       | 8173/31509 [00:14<00:34, 683.10it/s]","\rparsing log, completed traces ::  26%|##6       | 8244/31509 [00:14<00:33, 688.93it/s]","\rparsing log, completed traces ::  26%|##6       | 8315/31509 [00:14<00:33, 694.08it/s]","\rparsing log, completed traces ::  27%|##6       | 8385/31509 [00:14<00:34, 679.80it/s]","\rparsing log, completed traces ::  27%|##6       | 8454/31509 [00:15<00:34, 668.61it/s]","\rparsing log, completed traces ::  27%|##7       | 8524/31509 [00:15<00:34, 675.69it/s]","\rparsing log, completed traces ::  27%|##7       | 8592/31509 [00:15<00:34, 662.05it/s]","\rparsing log, completed traces ::  27%|##7       | 8659/31509 [00:15<01:22, 276.70it/s]","\rparsing log, completed traces ::  28%|##7       | 8727/31509 [00:15<01:07, 335.73it/s]","\rparsing log, completed traces ::  28%|##7       | 8796/31509 [00:15<00:57, 396.48it/s]","\rparsing log, completed traces ::  28%|##8       | 8864/31509 [00:16<00:50, 451.86it/s]","\rparsing log, completed traces ::  28%|##8       | 8927/31509 [00:16<00:46, 490.34it/s]","\rparsing log, completed traces ::  29%|##8       | 8994/31509 [00:16<00:42, 530.94it/s]","\rparsing log, completed traces ::  29%|##8       | 9058/31509 [00:16<00:41, 545.29it/s]","\rparsing log, completed traces ::  29%|##8       | 9125/31509 [00:16<00:38, 577.64it/s]","\rparsing log, completed traces ::  29%|##9       | 9193/31509 [00:16<00:37, 601.48it/s]","\rparsing log, completed traces ::  29%|##9       | 9258/31509 [00:16<00:36, 601.96it/s]","\rparsing log, completed traces ::  30%|##9       | 9331/31509 [00:16<00:35, 633.40it/s]","\rparsing log, completed traces ::  30%|##9       | 9397/31509 [00:16<00:35, 631.69it/s]","\rparsing log, completed traces ::  30%|###       | 9462/31509 [00:17<00:34, 634.38it/s]","\rparsing log, completed traces ::  30%|###       | 9529/31509 [00:17<00:34, 644.56it/s]","\rparsing log, completed traces ::  30%|###       | 9595/31509 [00:17<00:34, 635.32it/s]","\rparsing log, completed traces ::  31%|###       | 9662/31509 [00:17<00:33, 644.85it/s]","\rparsing log, completed traces ::  31%|###       | 9727/31509 [00:17<00:34, 633.58it/s]","\rparsing log, completed traces ::  31%|###1      | 9795/31509 [00:17<00:33, 644.37it/s]","\rparsing log, completed traces ::  31%|###1      | 9860/31509 [00:17<00:33, 639.99it/s]","\rparsing log, completed traces ::  31%|###1      | 9925/31509 [00:17<00:34, 634.32it/s]","\rparsing log, completed traces ::  32%|###1      | 9989/31509 [00:17<00:34, 631.49it/s]","\rparsing log, completed traces ::  32%|###1      | 10053/31509 [00:17<00:34, 626.35it/s]","\rparsing log, completed traces ::  32%|###2      | 10116/31509 [00:18<00:35, 604.08it/s]","\rparsing log, completed traces ::  32%|###2      | 10178/31509 [00:18<00:35, 608.48it/s]","\rparsing log, completed traces ::  33%|###2      | 10241/31509 [00:18<00:34, 613.53it/s]","\rparsing log, completed traces ::  33%|###2      | 10303/31509 [00:18<00:34, 612.44it/s]","\rparsing log, completed traces ::  33%|###2      | 10365/31509 [00:18<00:36, 586.91it/s]","\rparsing log, completed traces ::  33%|###3      | 10428/31509 [00:18<00:35, 598.58it/s]","\rparsing log, completed traces ::  33%|###3      | 10489/31509 [00:18<00:35, 593.54it/s]","\rparsing log, completed traces ::  33%|###3      | 10549/31509 [00:19<01:26, 242.78it/s]","\rparsing log, completed traces ::  34%|###3      | 10612/31509 [00:19<01:10, 296.91it/s]","\rparsing log, completed traces ::  34%|###3      | 10673/31509 [00:19<00:59, 349.63it/s]","\rparsing log, completed traces ::  34%|###4      | 10736/31509 [00:19<00:51, 403.24it/s]","\rparsing log, completed traces ::  34%|###4      | 10792/31509 [00:19<00:47, 434.42it/s]","\rparsing log, completed traces ::  34%|###4      | 10848/31509 [00:19<00:44, 460.87it/s]","\rparsing log, completed traces ::  35%|###4      | 10905/31509 [00:19<00:42, 487.54it/s]","\rparsing log, completed traces ::  35%|###4      | 10967/31509 [00:20<00:39, 520.45it/s]","\rparsing log, completed traces ::  35%|###5      | 11033/31509 [00:20<00:36, 555.33it/s]","\rparsing log, completed traces ::  35%|###5      | 11099/31509 [00:20<00:35, 581.99it/s]","\rparsing log, completed traces ::  35%|###5      | 11165/31509 [00:20<00:33, 603.06it/s]","\rparsing log, completed traces ::  36%|###5      | 11230/31509 [00:20<00:32, 615.97it/s]","\rparsing log, completed traces ::  36%|###5      | 11296/31509 [00:20<00:32, 626.79it/s]","\rparsing log, completed traces ::  36%|###6      | 11360/31509 [00:20<00:32, 616.56it/s]","\rparsing log, completed traces ::  36%|###6      | 11424/31509 [00:20<00:32, 622.26it/s]","\rparsing log, completed traces ::  36%|###6      | 11489/31509 [00:20<00:31, 627.19it/s]","\rparsing log, completed traces ::  37%|###6      | 11553/31509 [00:20<00:32, 614.18it/s]","\rparsing log, completed traces ::  37%|###6      | 11619/31509 [00:21<00:31, 625.10it/s]","\rparsing log, completed traces ::  37%|###7      | 11682/31509 [00:21<00:31, 625.05it/s]","\rparsing log, completed traces ::  37%|###7      | 11747/31509 [00:21<00:31, 631.61it/s]","\rparsing log, completed traces ::  37%|###7      | 11814/31509 [00:21<00:30, 641.93it/s]","\rparsing log, completed traces ::  38%|###7      | 11883/31509 [00:21<00:29, 655.11it/s]","\rparsing log, completed traces ::  38%|###7      | 11954/31509 [00:21<00:29, 670.93it/s]","\rparsing log, completed traces ::  38%|###8      | 12022/31509 [00:21<00:29, 658.60it/s]","\rparsing log, completed traces ::  38%|###8      | 12088/31509 [00:21<00:29, 658.44it/s]","\rparsing log, completed traces ::  39%|###8      | 12154/31509 [00:21<00:29, 654.44it/s]","\rparsing log, completed traces ::  39%|###8      | 12223/31509 [00:21<00:29, 664.52it/s]","\rparsing log, completed traces ::  39%|###9      | 12290/31509 [00:22<00:29, 657.44it/s]","\rparsing log, completed traces ::  39%|###9      | 12357/31509 [00:22<00:29, 658.87it/s]","\rparsing log, completed traces ::  39%|###9      | 12427/31509 [00:22<00:28, 666.02it/s]","\rparsing log, completed traces ::  40%|###9      | 12494/31509 [00:22<00:28, 663.30it/s]","\rparsing log, completed traces ::  40%|###9      | 12561/31509 [00:22<00:28, 654.08it/s]","\rparsing log, completed traces ::  40%|####      | 12629/31509 [00:22<00:28, 659.65it/s]","\rparsing log, completed traces ::  40%|####      | 12695/31509 [00:23<01:17, 242.23it/s]","\rparsing log, completed traces ::  41%|####      | 12762/31509 [00:23<01:02, 298.89it/s]","\rparsing log, completed traces ::  41%|####      | 12828/31509 [00:23<00:52, 356.41it/s]","\rparsing log, completed traces ::  41%|####      | 12892/31509 [00:23<00:45, 408.95it/s]","\rparsing log, completed traces ::  41%|####1     | 12954/31509 [00:23<00:40, 452.87it/s]","\rparsing log, completed traces ::  41%|####1     | 13024/31509 [00:23<00:36, 508.21it/s]","\rparsing log, completed traces ::  42%|####1     | 13101/31509 [00:23<00:32, 571.93it/s]","\rparsing log, completed traces ::  42%|####1     | 13168/31509 [00:23<00:31, 585.32it/s]","\rparsing log, completed traces ::  42%|####2     | 13234/31509 [00:24<00:30, 601.05it/s]","\rparsing log, completed traces ::  42%|####2     | 13300/31509 [00:24<00:29, 611.27it/s]","\rparsing log, completed traces ::  42%|####2     | 13366/31509 [00:24<00:29, 624.88it/s]","\rparsing log, completed traces ::  43%|####2     | 13433/31509 [00:24<00:28, 630.95it/s]","\rparsing log, completed traces ::  43%|####2     | 13498/31509 [00:24<00:28, 636.26it/s]","\rparsing log, completed traces ::  43%|####3     | 13563/31509 [00:24<00:29, 615.02it/s]","\rparsing log, completed traces ::  43%|####3     | 13626/31509 [00:24<00:29, 613.49it/s]","\rparsing log, completed traces ::  43%|####3     | 13690/31509 [00:24<00:28, 620.98it/s]","\rparsing log, completed traces ::  44%|####3     | 13759/31509 [00:24<00:27, 638.82it/s]","\rparsing log, completed traces ::  44%|####3     | 13824/31509 [00:24<00:27, 638.79it/s]","\rparsing log, completed traces ::  44%|####4     | 13889/31509 [00:25<00:28, 626.10it/s]","\rparsing log, completed traces ::  44%|####4     | 13955/31509 [00:25<00:27, 634.83it/s]","\rparsing log, completed traces ::  44%|####4     | 14020/31509 [00:25<00:27, 636.55it/s]","\rparsing log, completed traces ::  45%|####4     | 14089/31509 [00:25<00:26, 645.65it/s]","\rparsing log, completed traces ::  45%|####4     | 14159/31509 [00:25<00:26, 660.13it/s]","\rparsing log, completed traces ::  45%|####5     | 14226/31509 [00:25<00:26, 657.47it/s]","\rparsing log, completed traces ::  45%|####5     | 14293/31509 [00:25<00:26, 660.83it/s]","\rparsing log, completed traces ::  46%|####5     | 14364/31509 [00:25<00:25, 673.95it/s]","\rparsing log, completed traces ::  46%|####5     | 14433/31509 [00:25<00:25, 675.59it/s]","\rparsing log, completed traces ::  46%|####6     | 14501/31509 [00:26<00:25, 676.20it/s]","\rparsing log, completed traces ::  46%|####6     | 14569/31509 [00:26<00:26, 651.28it/s]","\rparsing log, completed traces ::  46%|####6     | 14635/31509 [00:26<00:26, 639.50it/s]","\rparsing log, completed traces ::  47%|####6     | 14702/31509 [00:26<00:25, 647.22it/s]","\rparsing log, completed traces ::  47%|####6     | 14767/31509 [00:26<00:25, 646.34it/s]","\rparsing log, completed traces ::  47%|####7     | 14838/31509 [00:26<00:25, 662.66it/s]","\rparsing log, completed traces ::  47%|####7     | 14905/31509 [00:26<00:25, 657.93it/s]","\rparsing log, completed traces ::  48%|####7     | 14976/31509 [00:26<00:24, 672.37it/s]","\rparsing log, completed traces ::  48%|####7     | 15044/31509 [00:26<00:24, 671.13it/s]","\rparsing log, completed traces ::  48%|####7     | 15116/31509 [00:26<00:23, 684.23it/s]","\rparsing log, completed traces ::  48%|####8     | 15185/31509 [00:27<01:09, 235.31it/s]","\rparsing log, completed traces ::  48%|####8     | 15253/31509 [00:27<00:55, 291.37it/s]","\rparsing log, completed traces ::  49%|####8     | 15326/31509 [00:27<00:45, 358.23it/s]","\rparsing log, completed traces ::  49%|####8     | 15396/31509 [00:27<00:38, 419.58it/s]","\rparsing log, completed traces ::  49%|####9     | 15461/31509 [00:28<00:34, 465.92it/s]","\rparsing log, completed traces ::  49%|####9     | 15527/31509 [00:28<00:31, 508.48it/s]","\rparsing log, completed traces ::  50%|####9     | 15597/31509 [00:28<00:28, 554.71it/s]","\rparsing log, completed traces ::  50%|####9     | 15671/31509 [00:28<00:26, 599.88it/s]","\rparsing log, completed traces ::  50%|####9     | 15739/31509 [00:28<00:25, 609.26it/s]","\rparsing log, completed traces ::  50%|#####     | 15807/31509 [00:28<00:25, 626.58it/s]","\rparsing log, completed traces ::  50%|#####     | 15880/31509 [00:28<00:23, 653.28it/s]","\rparsing log, completed traces ::  51%|#####     | 15949/31509 [00:28<00:23, 657.42it/s]","\rparsing log, completed traces ::  51%|#####     | 16017/31509 [00:28<00:23, 661.00it/s]","\rparsing log, completed traces ::  51%|#####1    | 16090/31509 [00:29<00:22, 678.81it/s]","\rparsing log, completed traces ::  51%|#####1    | 16165/31509 [00:29<00:21, 698.52it/s]","\rparsing log, completed traces ::  52%|#####1    | 16237/31509 [00:29<00:21, 703.69it/s]","\rparsing log, completed traces ::  52%|#####1    | 16309/31509 [00:29<00:21, 707.40it/s]","\rparsing log, completed traces ::  52%|#####1    | 16381/31509 [00:29<00:22, 676.29it/s]","\rparsing log, completed traces ::  52%|#####2    | 16450/31509 [00:29<00:22, 668.71it/s]","\rparsing log, completed traces ::  52%|#####2    | 16518/31509 [00:29<00:23, 646.38it/s]","\rparsing log, completed traces ::  53%|#####2    | 16589/31509 [00:29<00:22, 661.77it/s]","\rparsing log, completed traces ::  53%|#####2    | 16662/31509 [00:29<00:21, 680.27it/s]","\rparsing log, completed traces ::  53%|#####3    | 16731/31509 [00:29<00:21, 681.84it/s]","\rparsing log, completed traces ::  53%|#####3    | 16800/31509 [00:30<00:21, 672.66it/s]","\rparsing log, completed traces ::  54%|#####3    | 16868/31509 [00:30<00:21, 674.01it/s]","\rparsing log, completed traces ::  54%|#####3    | 16938/31509 [00:30<00:21, 681.07it/s]","\rparsing log, completed traces ::  54%|#####3    | 17007/31509 [00:30<00:21, 682.77it/s]","\rparsing log, completed traces ::  54%|#####4    | 17086/31509 [00:30<00:20, 714.31it/s]","\rparsing log, completed traces ::  54%|#####4    | 17158/31509 [00:30<00:20, 705.26it/s]","\rparsing log, completed traces ::  55%|#####4    | 17229/31509 [00:30<00:20, 687.89it/s]","\rparsing log, completed traces ::  55%|#####4    | 17301/31509 [00:30<00:20, 694.82it/s]","\rparsing log, completed traces ::  55%|#####5    | 17371/31509 [00:30<00:20, 689.95it/s]","\rparsing log, completed traces ::  55%|#####5    | 17447/31509 [00:30<00:19, 708.10it/s]","\rparsing log, completed traces ::  56%|#####5    | 17518/31509 [00:31<00:20, 676.68it/s]","\rparsing log, completed traces ::  56%|#####5    | 17591/31509 [00:31<00:20, 690.78it/s]","\rparsing log, completed traces ::  56%|#####6    | 17668/31509 [00:31<00:19, 713.32it/s]","\rparsing log, completed traces ::  56%|#####6    | 17740/31509 [00:31<00:19, 697.69it/s]","\rparsing log, completed traces ::  57%|#####6    | 17812/31509 [00:31<00:19, 702.37it/s]","\rparsing log, completed traces ::  57%|#####6    | 17884/31509 [00:31<00:19, 706.37it/s]","\rparsing log, completed traces ::  57%|#####6    | 17957/31509 [00:31<00:19, 712.83it/s]","\rparsing log, completed traces ::  57%|#####7    | 18031/31509 [00:31<00:18, 720.36it/s]","\rparsing log, completed traces ::  57%|#####7    | 18104/31509 [00:31<00:18, 718.57it/s]","\rparsing log, completed traces ::  58%|#####7    | 18176/31509 [00:32<00:57, 232.44it/s]","\rparsing log, completed traces ::  58%|#####7    | 18245/31509 [00:32<00:46, 287.28it/s]","\rparsing log, completed traces ::  58%|#####8    | 18312/31509 [00:32<00:38, 342.59it/s]","\rparsing log, completed traces ::  58%|#####8    | 18386/31509 [00:33<00:31, 411.26it/s]","\rparsing log, completed traces ::  59%|#####8    | 18457/31509 [00:33<00:27, 470.25it/s]","\rparsing log, completed traces ::  59%|#####8    | 18524/31509 [00:33<00:25, 512.80it/s]","\rparsing log, completed traces ::  59%|#####9    | 18591/31509 [00:33<00:23, 545.10it/s]","\rparsing log, completed traces ::  59%|#####9    | 18669/31509 [00:33<00:21, 603.89it/s]","\rparsing log, completed traces ::  59%|#####9    | 18744/31509 [00:33<00:19, 639.82it/s]","\rparsing log, completed traces ::  60%|#####9    | 18815/31509 [00:33<00:19, 647.08it/s]","\rparsing log, completed traces ::  60%|#####9    | 18890/31509 [00:33<00:18, 674.37it/s]","\rparsing log, completed traces ::  60%|######    | 18962/31509 [00:33<00:18, 684.78it/s]","\rparsing log, completed traces ::  60%|######    | 19033/31509 [00:33<00:18, 690.46it/s]","\rparsing log, completed traces ::  61%|######    | 19106/31509 [00:34<00:17, 699.21it/s]","\rparsing log, completed traces ::  61%|######    | 19178/31509 [00:34<00:17, 704.07it/s]","\rparsing log, completed traces ::  61%|######1   | 19250/31509 [00:34<00:17, 696.35it/s]","\rparsing log, completed traces ::  61%|######1   | 19321/31509 [00:34<00:17, 696.04it/s]","\rparsing log, completed traces ::  62%|######1   | 19397/31509 [00:34<00:16, 712.52it/s]","\rparsing log, completed traces ::  62%|######1   | 19469/31509 [00:34<00:16, 713.14it/s]","\rparsing log, completed traces ::  62%|######2   | 19541/31509 [00:34<00:16, 709.39it/s]","\rparsing log, completed traces ::  62%|######2   | 19613/31509 [00:34<00:16, 709.41it/s]","\rparsing log, completed traces ::  62%|######2   | 19685/31509 [00:34<00:17, 690.97it/s]","\rparsing log, completed traces ::  63%|######2   | 19756/31509 [00:34<00:16, 694.33it/s]","\rparsing log, completed traces ::  63%|######2   | 19826/31509 [00:35<00:16, 688.28it/s]","\rparsing log, completed traces ::  63%|######3   | 19895/31509 [00:35<00:17, 677.28it/s]","\rparsing log, completed traces ::  63%|######3   | 19963/31509 [00:35<00:17, 678.03it/s]","\rparsing log, completed traces ::  64%|######3   | 20031/31509 [00:35<00:17, 674.20it/s]","\rparsing log, completed traces ::  64%|######3   | 20099/31509 [00:35<00:17, 666.38it/s]","\rparsing log, completed traces ::  64%|######4   | 20168/31509 [00:35<00:16, 672.45it/s]","\rparsing log, completed traces ::  64%|######4   | 20242/31509 [00:35<00:16, 690.73it/s]","\rparsing log, completed traces ::  64%|######4   | 20313/31509 [00:35<00:16, 696.31it/s]","\rparsing log, completed traces ::  65%|######4   | 20383/31509 [00:35<00:15, 696.86it/s]","\rparsing log, completed traces ::  65%|######4   | 20462/31509 [00:35<00:15, 723.08it/s]","\rparsing log, completed traces ::  65%|######5   | 20535/31509 [00:36<00:15, 717.54it/s]","\rparsing log, completed traces ::  65%|######5   | 20608/31509 [00:36<00:15, 720.38it/s]","\rparsing log, completed traces ::  66%|######5   | 20683/31509 [00:36<00:14, 728.11it/s]","\rparsing log, completed traces ::  66%|######5   | 20758/31509 [00:36<00:14, 731.82it/s]","\rparsing log, completed traces ::  66%|######6   | 20832/31509 [00:36<00:15, 709.99it/s]","\rparsing log, completed traces ::  66%|######6   | 20904/31509 [00:36<00:14, 712.72it/s]","\rparsing log, completed traces ::  67%|######6   | 20976/31509 [00:36<00:15, 698.20it/s]","\rparsing log, completed traces ::  67%|######6   | 21046/31509 [00:36<00:15, 685.38it/s]","\rparsing log, completed traces ::  67%|######7   | 21115/31509 [00:36<00:15, 684.55it/s]","\rparsing log, completed traces ::  67%|######7   | 21184/31509 [00:37<00:15, 680.27it/s]","\rparsing log, completed traces ::  67%|######7   | 21259/31509 [00:37<00:14, 700.21it/s]","\rparsing log, completed traces ::  68%|######7   | 21330/31509 [00:37<00:14, 701.30it/s]","\rparsing log, completed traces ::  68%|######7   | 21401/31509 [00:37<00:14, 690.64it/s]","\rparsing log, completed traces ::  68%|######8   | 21471/31509 [00:38<00:48, 205.81it/s]","\rparsing log, completed traces ::  68%|######8   | 21545/31509 [00:38<00:37, 264.60it/s]","\rparsing log, completed traces ::  69%|######8   | 21616/31509 [00:38<00:30, 325.03it/s]","\rparsing log, completed traces ::  69%|######8   | 21680/31509 [00:38<00:26, 375.70it/s]","\rparsing log, completed traces ::  69%|######9   | 21761/31509 [00:38<00:21, 457.24it/s]","\rparsing log, completed traces ::  69%|######9   | 21835/31509 [00:38<00:18, 516.88it/s]","\rparsing log, completed traces ::  70%|######9   | 21912/31509 [00:38<00:16, 575.24it/s]","\rparsing log, completed traces ::  70%|######9   | 21984/31509 [00:38<00:15, 608.17it/s]","\rparsing log, completed traces ::  70%|#######   | 22061/31509 [00:39<00:14, 649.44it/s]","\rparsing log, completed traces ::  70%|#######   | 22136/31509 [00:39<00:13, 674.59it/s]","\rparsing log, completed traces ::  70%|#######   | 22210/31509 [00:39<00:13, 683.60it/s]","\rparsing log, completed traces ::  71%|#######   | 22284/31509 [00:39<00:13, 694.44it/s]","\rparsing log, completed traces ::  71%|#######   | 22357/31509 [00:39<00:13, 694.03it/s]","\rparsing log, completed traces ::  71%|#######1  | 22434/31509 [00:39<00:12, 714.53it/s]","\rparsing log, completed traces ::  71%|#######1  | 22510/31509 [00:39<00:12, 725.24it/s]","\rparsing log, completed traces ::  72%|#######1  | 22584/31509 [00:39<00:12, 711.45it/s]","\rparsing log, completed traces ::  72%|#######1  | 22656/31509 [00:39<00:12, 703.13it/s]","\rparsing log, completed traces ::  72%|#######2  | 22727/31509 [00:39<00:12, 699.37it/s]","\rparsing log, completed traces ::  72%|#######2  | 22800/31509 [00:40<00:12, 705.56it/s]","\rparsing log, completed traces ::  73%|#######2  | 22871/31509 [00:40<00:12, 706.33it/s]","\rparsing log, completed traces ::  73%|#######2  | 22942/31509 [00:40<00:12, 680.23it/s]","\rparsing log, completed traces ::  73%|#######3  | 23011/31509 [00:40<00:12, 676.07it/s]","\rparsing log, completed traces ::  73%|#######3  | 23081/31509 [00:40<00:12, 679.09it/s]","\rparsing log, completed traces ::  73%|#######3  | 23150/31509 [00:40<00:12, 677.73it/s]","\rparsing log, completed traces ::  74%|#######3  | 23218/31509 [00:40<00:12, 674.92it/s]","\rparsing log, completed traces ::  74%|#######3  | 23286/31509 [00:40<00:12, 668.84it/s]","\rparsing log, completed traces ::  74%|#######4  | 23353/31509 [00:40<00:12, 657.25it/s]","\rparsing log, completed traces ::  74%|#######4  | 23421/31509 [00:41<00:12, 660.96it/s]","\rparsing log, completed traces ::  75%|#######4  | 23488/31509 [00:41<00:12, 662.87it/s]","\rparsing log, completed traces ::  75%|#######4  | 23558/31509 [00:41<00:11, 673.12it/s]","\rparsing log, completed traces ::  75%|#######4  | 23629/31509 [00:41<00:11, 681.20it/s]","\rparsing log, completed traces ::  75%|#######5  | 23698/31509 [00:41<00:11, 678.52it/s]","\rparsing log, completed traces ::  75%|#######5  | 23770/31509 [00:41<00:11, 689.53it/s]","\rparsing log, completed traces ::  76%|#######5  | 23839/31509 [00:41<00:11, 677.59it/s]","\rparsing log, completed traces ::  76%|#######5  | 23908/31509 [00:41<00:11, 679.41it/s]","\rparsing log, completed traces ::  76%|#######6  | 23976/31509 [00:41<00:11, 678.21it/s]","\rparsing log, completed traces ::  76%|#######6  | 24044/31509 [00:41<00:11, 674.94it/s]","\rparsing log, completed traces ::  77%|#######6  | 24115/31509 [00:42<00:10, 684.79it/s]","\rparsing log, completed traces ::  77%|#######6  | 24184/31509 [00:42<00:10, 685.84it/s]","\rparsing log, completed traces ::  77%|#######6  | 24253/31509 [00:42<00:10, 677.10it/s]","\rparsing log, completed traces ::  77%|#######7  | 24326/31509 [00:42<00:10, 691.57it/s]","\rparsing log, completed traces ::  77%|#######7  | 24397/31509 [00:42<00:10, 696.18it/s]","\rparsing log, completed traces ::  78%|#######7  | 24467/31509 [00:42<00:10, 691.69it/s]","\rparsing log, completed traces ::  78%|#######7  | 24537/31509 [00:42<00:10, 692.39it/s]","\rparsing log, completed traces ::  78%|#######8  | 24617/31509 [00:42<00:09, 721.45it/s]","\rparsing log, completed traces ::  78%|#######8  | 24690/31509 [00:42<00:09, 707.99it/s]","\rparsing log, completed traces ::  79%|#######8  | 24761/31509 [00:42<00:09, 707.89it/s]","\rparsing log, completed traces ::  79%|#######8  | 24835/31509 [00:43<00:09, 714.27it/s]","\rparsing log, completed traces ::  79%|#######9  | 24907/31509 [00:43<00:09, 713.71it/s]","\rparsing log, completed traces ::  79%|#######9  | 24983/31509 [00:43<00:09, 723.21it/s]","\rparsing log, completed traces ::  80%|#######9  | 25056/31509 [00:43<00:09, 705.02it/s]","\rparsing log, completed traces ::  80%|#######9  | 25127/31509 [00:43<00:09, 689.77it/s]","\rparsing log, completed traces ::  80%|#######9  | 25197/31509 [00:44<00:33, 190.52it/s]","\rparsing log, completed traces ::  80%|########  | 25263/31509 [00:44<00:26, 238.35it/s]","\rparsing log, completed traces ::  80%|########  | 25322/31509 [00:44<00:21, 282.56it/s]","\rparsing log, completed traces ::  81%|########  | 25396/31509 [00:44<00:17, 352.47it/s]","\rparsing log, completed traces ::  81%|########  | 25463/31509 [00:44<00:14, 409.20it/s]","\rparsing log, completed traces ::  81%|########1 | 25531/31509 [00:44<00:12, 464.20it/s]","\rparsing log, completed traces ::  81%|########1 | 25603/31509 [00:45<00:11, 519.40it/s]","\rparsing log, completed traces ::  81%|########1 | 25670/31509 [00:45<00:10, 552.76it/s]","\rparsing log, completed traces ::  82%|########1 | 25738/31509 [00:45<00:09, 583.98it/s]","\rparsing log, completed traces ::  82%|########1 | 25806/31509 [00:45<00:09, 608.08it/s]","\rparsing log, completed traces ::  82%|########2 | 25873/31509 [00:45<00:09, 624.51it/s]","\rparsing log, completed traces ::  82%|########2 | 25943/31509 [00:45<00:08, 643.49it/s]","\rparsing log, completed traces ::  83%|########2 | 26011/31509 [00:45<00:08, 640.79it/s]","\rparsing log, completed traces ::  83%|########2 | 26078/31509 [00:45<00:08, 645.91it/s]","\rparsing log, completed traces ::  83%|########2 | 26148/31509 [00:45<00:08, 652.58it/s]","\rparsing log, completed traces ::  83%|########3 | 26220/31509 [00:46<00:07, 672.05it/s]","\rparsing log, completed traces ::  83%|########3 | 26292/31509 [00:46<00:07, 684.06it/s]","\rparsing log, completed traces ::  84%|########3 | 26364/31509 [00:46<00:07, 694.53it/s]","\rparsing log, completed traces ::  84%|########3 | 26434/31509 [00:46<00:07, 666.59it/s]","\rparsing log, completed traces ::  84%|########4 | 26502/31509 [00:46<00:07, 653.29it/s]","\rparsing log, completed traces ::  84%|########4 | 26569/31509 [00:46<00:07, 657.68it/s]","\rparsing log, completed traces ::  85%|########4 | 26636/31509 [00:46<00:07, 654.91it/s]","\rparsing log, completed traces ::  85%|########4 | 26703/31509 [00:46<00:07, 658.51it/s]","\rparsing log, completed traces ::  85%|########4 | 26775/31509 [00:46<00:07, 673.59it/s]","\rparsing log, completed traces ::  85%|########5 | 26844/31509 [00:46<00:06, 676.29it/s]","\rparsing log, completed traces ::  85%|########5 | 26912/31509 [00:47<00:06, 666.20it/s]","\rparsing log, completed traces ::  86%|########5 | 26980/31509 [00:47<00:06, 668.97it/s]","\rparsing log, completed traces ::  86%|########5 | 27047/31509 [00:47<00:06, 654.00it/s]","\rparsing log, completed traces ::  86%|########6 | 27116/31509 [00:47<00:06, 662.32it/s]","\rparsing log, completed traces ::  86%|########6 | 27186/31509 [00:47<00:06, 672.06it/s]","\rparsing log, completed traces ::  87%|########6 | 27260/31509 [00:47<00:06, 691.06it/s]","\rparsing log, completed traces ::  87%|########6 | 27330/31509 [00:47<00:06, 667.46it/s]","\rparsing log, completed traces ::  87%|########6 | 27399/31509 [00:47<00:06, 672.94it/s]","\rparsing log, completed traces ::  87%|########7 | 27468/31509 [00:47<00:05, 676.07it/s]","\rparsing log, completed traces ::  87%|########7 | 27536/31509 [00:47<00:05, 669.67it/s]","\rparsing log, completed traces ::  88%|########7 | 27610/31509 [00:48<00:05, 686.40it/s]","\rparsing log, completed traces ::  88%|########7 | 27681/31509 [00:48<00:05, 693.03it/s]","\rparsing log, completed traces ::  88%|########8 | 27756/31509 [00:48<00:05, 707.49it/s]","\rparsing log, completed traces ::  88%|########8 | 27827/31509 [00:48<00:05, 690.73it/s]","\rparsing log, completed traces ::  89%|########8 | 27897/31509 [00:48<00:05, 681.16it/s]","\rparsing log, completed traces ::  89%|########8 | 27966/31509 [00:48<00:05, 672.92it/s]","\rparsing log, completed traces ::  89%|########8 | 28038/31509 [00:48<00:05, 684.26it/s]","\rparsing log, completed traces ::  89%|########9 | 28107/31509 [00:48<00:05, 659.72it/s]","\rparsing log, completed traces ::  89%|########9 | 28174/31509 [00:48<00:05, 661.70it/s]","\rparsing log, completed traces ::  90%|########9 | 28246/31509 [00:49<00:04, 678.04it/s]","\rparsing log, completed traces ::  90%|########9 | 28314/31509 [00:49<00:04, 670.52it/s]","\rparsing log, completed traces ::  90%|######### | 28385/31509 [00:49<00:04, 680.46it/s]","\rparsing log, completed traces ::  90%|######### | 28454/31509 [00:49<00:04, 668.87it/s]","\rparsing log, completed traces ::  91%|######### | 28526/31509 [00:49<00:04, 683.61it/s]","\rparsing log, completed traces ::  91%|######### | 28599/31509 [00:49<00:04, 696.73it/s]","\rparsing log, completed traces ::  91%|######### | 28669/31509 [00:49<00:04, 691.31it/s]","\rparsing log, completed traces ::  91%|#########1| 28739/31509 [00:49<00:04, 654.40it/s]","\rparsing log, completed traces ::  91%|#########1| 28805/31509 [00:49<00:04, 643.44it/s]","\rparsing log, completed traces ::  92%|#########1| 28870/31509 [00:49<00:04, 628.60it/s]","\rparsing log, completed traces ::  92%|#########1| 28940/31509 [00:50<00:03, 646.19it/s]","\rparsing log, completed traces ::  92%|#########2| 29010/31509 [00:50<00:03, 659.17it/s]","\rparsing log, completed traces ::  92%|#########2| 29077/31509 [00:50<00:03, 647.55it/s]","\rparsing log, completed traces ::  92%|#########2| 29142/31509 [00:50<00:03, 647.64it/s]","\rparsing log, completed traces ::  93%|#########2| 29214/31509 [00:50<00:03, 668.05it/s]","\rparsing log, completed traces ::  93%|#########2| 29281/31509 [00:51<00:13, 168.66it/s]","\rparsing log, completed traces ::  93%|#########3| 29351/31509 [00:51<00:09, 219.39it/s]","\rparsing log, completed traces ::  93%|#########3| 29414/31509 [00:51<00:07, 268.53it/s]","\rparsing log, completed traces ::  94%|#########3| 29476/31509 [00:51<00:06, 319.31it/s]","\rparsing log, completed traces ::  94%|#########3| 29535/31509 [00:51<00:05, 363.59it/s]","\rparsing log, completed traces ::  94%|#########3| 29605/31509 [00:52<00:04, 428.59it/s]","\rparsing log, completed traces ::  94%|#########4| 29675/31509 [00:52<00:03, 486.58it/s]","\rparsing log, completed traces ::  94%|#########4| 29743/31509 [00:52<00:03, 531.70it/s]","\rparsing log, completed traces ::  95%|#########4| 29808/31509 [00:52<00:03, 556.23it/s]","\rparsing log, completed traces ::  95%|#########4| 29876/31509 [00:52<00:02, 588.50it/s]","\rparsing log, completed traces ::  95%|#########5| 29944/31509 [00:52<00:02, 613.05it/s]","\rparsing log, completed traces ::  95%|#########5| 30012/31509 [00:52<00:02, 629.70it/s]","\rparsing log, completed traces ::  95%|#########5| 30079/31509 [00:52<00:02, 630.48it/s]","\rparsing log, completed traces ::  96%|#########5| 30152/31509 [00:52<00:02, 657.82it/s]","\rparsing log, completed traces ::  96%|#########5| 30223/31509 [00:53<00:01, 669.52it/s]","\rparsing log, completed traces ::  96%|#########6| 30292/31509 [00:53<00:01, 667.61it/s]","\rparsing log, completed traces ::  96%|#########6| 30366/31509 [00:53<00:01, 688.73it/s]","\rparsing log, completed traces ::  97%|#########6| 30436/31509 [00:53<00:01, 679.73it/s]","\rparsing log, completed traces ::  97%|#########6| 30505/31509 [00:53<00:01, 677.36it/s]","\rparsing log, completed traces ::  97%|#########7| 30580/31509 [00:53<00:01, 697.49it/s]","\rparsing log, completed traces ::  97%|#########7| 30651/31509 [00:53<00:01, 692.13it/s]","\rparsing log, completed traces ::  97%|#########7| 30721/31509 [00:53<00:01, 692.77it/s]","\rparsing log, completed traces ::  98%|#########7| 30791/31509 [00:53<00:01, 687.77it/s]","\rparsing log, completed traces ::  98%|#########7| 30861/31509 [00:53<00:00, 688.98it/s]","\rparsing log, completed traces ::  98%|#########8| 30930/31509 [00:54<00:00, 680.64it/s]","\rparsing log, completed traces ::  98%|#########8| 31001/31509 [00:54<00:00, 687.52it/s]","\rparsing log, completed traces ::  99%|#########8| 31072/31509 [00:54<00:00, 693.32it/s]","\rparsing log, completed traces ::  99%|#########8| 31143/31509 [00:54<00:00, 691.55it/s]","\rparsing log, completed traces ::  99%|#########9| 31219/31509 [00:54<00:00, 709.33it/s]","\rparsing log, completed traces ::  99%|#########9| 31290/31509 [00:54<00:00, 667.30it/s]","\rparsing log, completed traces :: 100%|#########9| 31358/31509 [00:54<00:00, 660.42it/s]","\rparsing log, completed traces :: 100%|#########9| 31429/31509 [00:54<00:00, 672.89it/s]","\rparsing log, completed traces :: 100%|#########9| 31497/31509 [00:54<00:00, 659.35it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:54<00:00, 574.00it/s]","\n","[data] Loading XES: /workspace/data/Road_Traffic_Fine_Management_Process.xes","\n","\rparsing log, completed traces ::   0%|          | 0/150370 [00:00<?, ?it/s]","\rparsing log, completed traces ::   1%|          | 756/150370 [00:00<00:19, 7540.12it/s]","\rparsing log, completed traces ::   1%|1         | 1574/150370 [00:00<00:18, 7878.00it/s]","\rparsing log, completed traces ::   2%|1         | 2383/150370 [00:00<00:18, 7973.92it/s]","\rparsing log, completed traces ::   2%|2         | 3181/150370 [00:00<00:18, 7938.32it/s]","\rparsing log, completed traces ::   3%|2         | 3984/150370 [00:00<00:18, 7971.20it/s]","\rparsing log, completed traces ::   3%|3         | 4782/150370 [00:00<00:18, 7960.41it/s]","\rparsing log, completed traces ::   4%|3         | 5585/150370 [00:00<00:18, 7982.21it/s]","\rparsing log, completed traces ::   4%|4         | 6384/150370 [00:00<00:18, 7932.17it/s]","\rparsing log, completed traces ::   5%|4         | 7178/150370 [00:01<00:30, 4653.86it/s]","\rparsing log, completed traces ::   5%|5         | 7969/150370 [00:01<00:26, 5326.03it/s]","\rparsing log, completed traces ::   6%|5         | 8805/150370 [00:01<00:23, 6005.56it/s]","\rparsing log, completed traces ::   6%|6         | 9643/150370 [00:01<00:21, 6572.97it/s]","\rparsing log, completed traces ::   7%|6         | 10437/150370 [00:01<00:20, 6926.75it/s]","\rparsing log, completed traces ::   7%|7         | 11243/150370 [00:01<00:19, 7232.11it/s]","\rparsing log, completed traces ::   8%|8         | 12051/150370 [00:01<00:18, 7468.03it/s]","\rparsing log, completed traces ::   9%|8         | 12839/150370 [00:01<00:18, 7561.81it/s]","\rparsing log, completed traces ::   9%|9         | 13639/150370 [00:01<00:17, 7687.41it/s]","\rparsing log, completed traces ::  10%|9         | 14429/150370 [00:02<00:17, 7728.79it/s]","\rparsing log, completed traces ::  10%|#         | 15228/150370 [00:02<00:17, 7803.92it/s]","\rparsing log, completed traces ::  11%|#         | 16069/150370 [00:02<00:16, 7981.77it/s]","\rparsing log, completed traces ::  11%|#1        | 16917/150370 [00:02<00:16, 8107.14it/s]","\rparsing log, completed traces ::  12%|#1        | 17734/150370 [00:02<00:30, 4411.47it/s]","\rparsing log, completed traces ::  12%|#2        | 18550/150370 [00:02<00:25, 5113.56it/s]","\rparsing log, completed traces ::  13%|#2        | 19351/150370 [00:02<00:22, 5724.88it/s]","\rparsing log, completed traces ::  13%|#3        | 20168/150370 [00:03<00:20, 6289.56it/s]","\rparsing log, completed traces ::  14%|#3        | 20981/150370 [00:03<00:19, 6746.04it/s]","\rparsing log, completed traces ::  15%|#4        | 21870/150370 [00:03<00:17, 7304.98it/s]","\rparsing log, completed traces ::  15%|#5        | 22750/150370 [00:03<00:16, 7711.89it/s]","\rparsing log, completed traces ::  16%|#5        | 23582/150370 [00:03<00:16, 7880.28it/s]","\rparsing log, completed traces ::  16%|#6        | 24411/150370 [00:03<00:16, 7456.14it/s]","\rparsing log, completed traces ::  17%|#6        | 25190/150370 [00:03<00:17, 7187.77it/s]","\rparsing log, completed traces ::  17%|#7        | 25933/150370 [00:03<00:17, 7035.78it/s]","\rparsing log, completed traces ::  18%|#7        | 26653/150370 [00:03<00:18, 6801.81it/s]","\rparsing log, completed traces ::  18%|#8        | 27345/150370 [00:03<00:18, 6750.51it/s]","\rparsing log, completed traces ::  19%|#8        | 28028/150370 [00:04<00:18, 6635.03it/s]","\rparsing log, completed traces ::  19%|#9        | 28697/150370 [00:04<00:33, 3599.90it/s]","\rparsing log, completed traces ::  20%|#9        | 29488/150370 [00:04<00:27, 4367.39it/s]","\rparsing log, completed traces ::  20%|##        | 30203/150370 [00:04<00:24, 4930.41it/s]","\rparsing log, completed traces ::  21%|##        | 30885/150370 [00:04<00:22, 5342.17it/s]","\rparsing log, completed traces ::  21%|##        | 31552/150370 [00:04<00:20, 5663.21it/s]","\rparsing log, completed traces ::  21%|##1       | 32240/150370 [00:05<00:19, 5973.69it/s]","\rparsing log, completed traces ::  22%|##1       | 32914/150370 [00:05<00:19, 6178.10it/s]","\rparsing log, completed traces ::  22%|##2       | 33596/150370 [00:05<00:18, 6356.09it/s]","\rparsing log, completed traces ::  23%|##2       | 34274/150370 [00:05<00:17, 6476.00it/s]","\rparsing log, completed traces ::  23%|##3       | 34948/150370 [00:05<00:17, 6551.28it/s]","\rparsing log, completed traces ::  24%|##3       | 35627/150370 [00:05<00:17, 6620.53it/s]","\rparsing log, completed traces ::  24%|##4       | 36315/150370 [00:05<00:17, 6695.42it/s]","\rparsing log, completed traces ::  25%|##4       | 37011/150370 [00:05<00:16, 6772.57it/s]","\rparsing log, completed traces ::  25%|##5       | 37705/150370 [00:05<00:16, 6822.06it/s]","\rparsing log, completed traces ::  26%|##5       | 38422/150370 [00:05<00:16, 6925.15it/s]","\rparsing log, completed traces ::  26%|##6       | 39140/150370 [00:06<00:15, 7001.03it/s]","\rparsing log, completed traces ::  27%|##6       | 39872/150370 [00:06<00:15, 7095.51it/s]","\rparsing log, completed traces ::  27%|##6       | 40584/150370 [00:06<00:32, 3370.22it/s]","\rparsing log, completed traces ::  27%|##7       | 41324/150370 [00:06<00:26, 4049.58it/s]","\rparsing log, completed traces ::  28%|##7       | 42083/150370 [00:06<00:22, 4736.01it/s]","\rparsing log, completed traces ::  28%|##8       | 42823/150370 [00:06<00:20, 5314.31it/s]","\rparsing log, completed traces ::  29%|##8       | 43587/150370 [00:06<00:18, 5864.14it/s]","\rparsing log, completed traces ::  29%|##9       | 44329/150370 [00:07<00:16, 6256.54it/s]","\rparsing log, completed traces ::  30%|##9       | 45055/150370 [00:07<00:16, 6522.64it/s]","\rparsing log, completed traces ::  30%|###       | 45794/150370 [00:07<00:15, 6760.03it/s]","\rparsing log, completed traces ::  31%|###       | 46519/150370 [00:07<00:15, 6859.68it/s]","\rparsing log, completed traces ::  31%|###1      | 47240/150370 [00:07<00:14, 6889.89it/s]","\rparsing log, completed traces ::  32%|###1      | 47983/150370 [00:07<00:14, 7045.35it/s]","\rparsing log, completed traces ::  32%|###2      | 48759/150370 [00:07<00:14, 7240.60it/s]","\rparsing log, completed traces ::  33%|###2      | 49571/150370 [00:07<00:13, 7485.40it/s]","\rparsing log, completed traces ::  33%|###3      | 50329/150370 [00:07<00:13, 7488.42it/s]","\rparsing log, completed traces ::  34%|###3      | 51085/150370 [00:07<00:13, 7470.02it/s]","\rparsing log, completed traces ::  34%|###4      | 51837/150370 [00:08<00:13, 7454.63it/s]","\rparsing log, completed traces ::  35%|###4      | 52586/150370 [00:08<00:13, 7437.26it/s]","\rparsing log, completed traces ::  35%|###5      | 53332/150370 [00:08<00:13, 7401.79it/s]","\rparsing log, completed traces ::  36%|###5      | 54086/150370 [00:08<00:12, 7442.41it/s]","\rparsing log, completed traces ::  37%|###6      | 54901/150370 [00:08<00:12, 7636.58it/s]","\rparsing log, completed traces ::  37%|###7      | 55666/150370 [00:09<00:28, 3381.56it/s]","\rparsing log, completed traces ::  38%|###7      | 56471/150370 [00:09<00:22, 4121.10it/s]","\rparsing log, completed traces ::  38%|###8      | 57253/150370 [00:09<00:19, 4802.65it/s]","\rparsing log, completed traces ::  39%|###8      | 58018/150370 [00:09<00:17, 5395.87it/s]","\rparsing log, completed traces ::  39%|###9      | 58754/150370 [00:09<00:15, 5838.03it/s]","\rparsing log, completed traces ::  40%|###9      | 59484/150370 [00:09<00:14, 6198.41it/s]","\rparsing log, completed traces ::  40%|####      | 60233/150370 [00:09<00:13, 6533.63it/s]","\rparsing log, completed traces ::  41%|####      | 61042/150370 [00:09<00:12, 6947.01it/s]","\rparsing log, completed traces ::  41%|####1     | 61861/150370 [00:09<00:12, 7276.59it/s]","\rparsing log, completed traces ::  42%|####1     | 62644/150370 [00:09<00:11, 7415.07it/s]","\rparsing log, completed traces ::  42%|####2     | 63416/150370 [00:10<00:11, 7418.75it/s]","\rparsing log, completed traces ::  43%|####2     | 64180/150370 [00:10<00:11, 7425.81it/s]","\rparsing log, completed traces ::  43%|####3     | 64941/150370 [00:10<00:11, 7477.64it/s]","\rparsing log, completed traces ::  44%|####3     | 65734/150370 [00:10<00:11, 7588.59it/s]","\rparsing log, completed traces ::  44%|####4     | 66501/150370 [00:10<00:11, 7579.05it/s]","\rparsing log, completed traces ::  45%|####4     | 67294/150370 [00:10<00:10, 7676.90it/s]","\rparsing log, completed traces ::  45%|####5     | 68150/150370 [00:10<00:10, 7938.84it/s]","\rparsing log, completed traces ::  46%|####5     | 68947/150370 [00:10<00:10, 7912.54it/s]","\rparsing log, completed traces ::  46%|####6     | 69741/150370 [00:10<00:10, 7858.45it/s]","\rparsing log, completed traces ::  47%|####6     | 70529/150370 [00:10<00:10, 7852.02it/s]","\rparsing log, completed traces ::  47%|####7     | 71316/150370 [00:11<00:10, 7856.81it/s]","\rparsing log, completed traces ::  48%|####7     | 72103/150370 [00:11<00:10, 7801.97it/s]","\rparsing log, completed traces ::  48%|####8     | 72884/150370 [00:11<00:09, 7782.54it/s]","\rparsing log, completed traces ::  49%|####8     | 73663/150370 [00:11<00:10, 7585.31it/s]","\rparsing log, completed traces ::  49%|####9     | 74423/150370 [00:11<00:23, 3175.71it/s]","\rparsing log, completed traces ::  50%|#####     | 75277/150370 [00:12<00:18, 3974.55it/s]","\rparsing log, completed traces ::  51%|#####     | 76078/150370 [00:12<00:15, 4675.56it/s]","\rparsing log, completed traces ::  51%|#####1    | 76904/150370 [00:12<00:13, 5392.60it/s]","\rparsing log, completed traces ::  52%|#####1    | 77673/150370 [00:12<00:12, 5902.63it/s]","\rparsing log, completed traces ::  52%|#####2    | 78440/150370 [00:12<00:11, 6327.53it/s]","\rparsing log, completed traces ::  53%|#####2    | 79231/150370 [00:12<00:10, 6732.76it/s]","\rparsing log, completed traces ::  53%|#####3    | 80009/150370 [00:12<00:10, 7012.33it/s]","\rparsing log, completed traces ::  54%|#####3    | 80780/150370 [00:12<00:09, 7189.82it/s]","\rparsing log, completed traces ::  54%|#####4    | 81557/150370 [00:12<00:09, 7353.58it/s]","\rparsing log, completed traces ::  55%|#####4    | 82397/150370 [00:12<00:08, 7640.34it/s]","\rparsing log, completed traces ::  55%|#####5    | 83229/150370 [00:13<00:08, 7837.85it/s]","\rparsing log, completed traces ::  56%|#####5    | 84032/150370 [00:13<00:08, 7870.22it/s]","\rparsing log, completed traces ::  56%|#####6    | 84832/150370 [00:13<00:08, 7845.08it/s]","\rparsing log, completed traces ::  57%|#####6    | 85626/150370 [00:13<00:08, 7814.49it/s]","\rparsing log, completed traces ::  57%|#####7    | 86428/150370 [00:13<00:08, 7874.38it/s]","\rparsing log, completed traces ::  58%|#####8    | 87237/150370 [00:13<00:07, 7937.92it/s]","\rparsing log, completed traces ::  59%|#####8    | 88037/150370 [00:13<00:07, 7955.33it/s]","\rparsing log, completed traces ::  59%|#####9    | 88845/150370 [00:13<00:07, 7992.33it/s]","\rparsing log, completed traces ::  60%|#####9    | 89664/150370 [00:13<00:07, 8050.16it/s]","\rparsing log, completed traces ::  60%|######    | 90477/150370 [00:13<00:07, 8072.24it/s]","\rparsing log, completed traces ::  61%|######    | 91287/150370 [00:14<00:07, 8078.44it/s]","\rparsing log, completed traces ::  61%|######1   | 92096/150370 [00:14<00:07, 8078.95it/s]","\rparsing log, completed traces ::  62%|######1   | 92905/150370 [00:14<00:07, 8035.58it/s]","\rparsing log, completed traces ::  62%|######2   | 93709/150370 [00:14<00:07, 8032.41it/s]","\rparsing log, completed traces ::  63%|######2   | 94535/150370 [00:14<00:06, 8099.94it/s]","\rparsing log, completed traces ::  63%|######3   | 95346/150370 [00:15<00:17, 3060.83it/s]","\rparsing log, completed traces ::  64%|######3   | 96127/150370 [00:15<00:14, 3720.60it/s]","\rparsing log, completed traces ::  64%|######4   | 96902/150370 [00:15<00:12, 4387.40it/s]","\rparsing log, completed traces ::  65%|######4   | 97674/150370 [00:15<00:10, 5024.51it/s]","\rparsing log, completed traces ::  65%|######5   | 98472/150370 [00:15<00:09, 5658.70it/s]","\rparsing log, completed traces ::  66%|######6   | 99273/150370 [00:15<00:08, 6209.69it/s]","\rparsing log, completed traces ::  67%|######6   | 100054/150370 [00:15<00:07, 6595.86it/s]","\rparsing log, completed traces ::  67%|######7   | 100857/150370 [00:15<00:07, 6971.46it/s]","\rparsing log, completed traces ::  68%|######7   | 101666/150370 [00:15<00:06, 7275.53it/s]","\rparsing log, completed traces ::  68%|######8   | 102471/150370 [00:15<00:06, 7488.80it/s]","\rparsing log, completed traces ::  69%|######8   | 103262/150370 [00:16<00:06, 7596.07it/s]","\rparsing log, completed traces ::  69%|######9   | 104052/150370 [00:16<00:06, 7665.41it/s]","\rparsing log, completed traces ::  70%|######9   | 104840/150370 [00:16<00:05, 7674.89it/s]","\rparsing log, completed traces ::  70%|#######   | 105623/150370 [00:16<00:05, 7717.10it/s]","\rparsing log, completed traces ::  71%|#######   | 106406/150370 [00:16<00:05, 7722.85it/s]","\rparsing log, completed traces ::  71%|#######1  | 107186/150370 [00:16<00:05, 7716.03it/s]","\rparsing log, completed traces ::  72%|#######1  | 107963/150370 [00:16<00:05, 7708.26it/s]","\rparsing log, completed traces ::  72%|#######2  | 108738/150370 [00:16<00:05, 7701.96it/s]","\rparsing log, completed traces ::  73%|#######2  | 109514/150370 [00:16<00:05, 7718.91it/s]","\rparsing log, completed traces ::  73%|#######3  | 110288/150370 [00:16<00:05, 7632.46it/s]","\rparsing log, completed traces ::  74%|#######3  | 111077/150370 [00:17<00:05, 7706.97it/s]","\rparsing log, completed traces ::  74%|#######4  | 111862/150370 [00:17<00:04, 7749.37it/s]","\rparsing log, completed traces ::  75%|#######4  | 112638/150370 [00:17<00:04, 7646.52it/s]","\rparsing log, completed traces ::  75%|#######5  | 113404/150370 [00:17<00:04, 7648.86it/s]","\rparsing log, completed traces ::  76%|#######6  | 114343/150370 [00:17<00:04, 8163.98it/s]","\rparsing log, completed traces ::  77%|#######6  | 115287/150370 [00:17<00:04, 8543.77it/s]","\rparsing log, completed traces ::  77%|#######7  | 116211/150370 [00:17<00:03, 8750.35it/s]","\rparsing log, completed traces ::  78%|#######7  | 117286/150370 [00:17<00:03, 9346.57it/s]","\rparsing log, completed traces ::  79%|#######8  | 118222/150370 [00:17<00:03, 8676.30it/s]","\rparsing log, completed traces ::  79%|#######9  | 119100/150370 [00:18<00:10, 3029.43it/s]","\rparsing log, completed traces ::  80%|#######9  | 119819/150370 [00:18<00:08, 3543.76it/s]","\rparsing log, completed traces ::  80%|########  | 120548/150370 [00:18<00:07, 4106.18it/s]","\rparsing log, completed traces ::  81%|########  | 121267/150370 [00:18<00:06, 4649.61it/s]","\rparsing log, completed traces ::  81%|########1 | 121997/150370 [00:19<00:05, 5182.76it/s]","\rparsing log, completed traces ::  82%|########1 | 122735/150370 [00:19<00:04, 5672.17it/s]","\rparsing log, completed traces ::  82%|########2 | 123464/150370 [00:19<00:04, 6063.48it/s]","\rparsing log, completed traces ::  83%|########2 | 124184/150370 [00:19<00:04, 6336.99it/s]","\rparsing log, completed traces ::  83%|########3 | 124902/150370 [00:19<00:03, 6542.32it/s]","\rparsing log, completed traces ::  84%|########3 | 125639/150370 [00:19<00:03, 6770.60it/s]","\rparsing log, completed traces ::  84%|########4 | 126371/150370 [00:19<00:03, 6923.42it/s]","\rparsing log, completed traces ::  85%|########4 | 127130/150370 [00:19<00:03, 7107.83it/s]","\rparsing log, completed traces ::  85%|########5 | 127873/150370 [00:19<00:03, 7201.43it/s]","\rparsing log, completed traces ::  86%|########5 | 128625/150370 [00:19<00:02, 7289.86it/s]","\rparsing log, completed traces ::  86%|########6 | 129381/150370 [00:20<00:02, 7356.36it/s]","\rparsing log, completed traces ::  87%|########6 | 130131/150370 [00:20<00:02, 7383.98it/s]","\rparsing log, completed traces ::  87%|########7 | 130892/150370 [00:20<00:02, 7435.82it/s]","\rparsing log, completed traces ::  88%|########7 | 131669/150370 [00:20<00:02, 7535.20it/s]","\rparsing log, completed traces ::  88%|########8 | 132449/150370 [00:20<00:02, 7611.94it/s]","\rparsing log, completed traces ::  89%|########8 | 133300/150370 [00:20<00:02, 7879.42it/s]","\rparsing log, completed traces ::  89%|########9 | 134090/150370 [00:20<00:02, 7797.94it/s]","\rparsing log, completed traces ::  90%|########9 | 134872/150370 [00:20<00:02, 7746.45it/s]","\rparsing log, completed traces ::  90%|######### | 135648/150370 [00:20<00:01, 7652.96it/s]","\rparsing log, completed traces ::  91%|######### | 136415/150370 [00:21<00:01, 7565.93it/s]","\rparsing log, completed traces ::  91%|#########1| 137173/150370 [00:21<00:01, 7498.47it/s]","\rparsing log, completed traces ::  92%|#########1| 137924/150370 [00:21<00:01, 7453.64it/s]","\rparsing log, completed traces ::  92%|#########2| 138678/150370 [00:21<00:01, 7477.18it/s]","\rparsing log, completed traces ::  93%|#########2| 139428/150370 [00:21<00:01, 7482.93it/s]","\rparsing log, completed traces ::  93%|#########3| 140177/150370 [00:21<00:01, 7478.24it/s]","\rparsing log, completed traces ::  94%|#########3| 140964/150370 [00:21<00:01, 7592.62it/s]","\rparsing log, completed traces ::  94%|#########4| 141743/150370 [00:21<00:01, 7650.41it/s]","\rparsing log, completed traces ::  95%|#########4| 142509/150370 [00:21<00:01, 7644.57it/s]","\rparsing log, completed traces ::  95%|#########5| 143274/150370 [00:21<00:00, 7629.44it/s]","\rparsing log, completed traces ::  96%|#########5| 144042/150370 [00:22<00:00, 7637.04it/s]","\rparsing log, completed traces ::  96%|#########6| 144806/150370 [00:22<00:00, 7591.06it/s]","\rparsing log, completed traces ::  97%|#########6| 145566/150370 [00:22<00:02, 2355.48it/s]","\rparsing log, completed traces ::  97%|#########7| 146154/150370 [00:23<00:01, 2766.47it/s]","\rparsing log, completed traces ::  98%|#########7| 146772/150370 [00:23<00:01, 3253.93it/s]","\rparsing log, completed traces ::  98%|#########8| 147375/150370 [00:23<00:00, 3724.03it/s]","\rparsing log, completed traces ::  98%|#########8| 147976/150370 [00:23<00:00, 4163.40it/s]","\rparsing log, completed traces ::  99%|#########8| 148614/150370 [00:23<00:00, 4645.39it/s]","\rparsing log, completed traces ::  99%|#########9| 149281/150370 [00:23<00:00, 5125.30it/s]","\rparsing log, completed traces :: 100%|#########9| 149993/150370 [00:23<00:00, 5628.65it/s]","","\rparsing log, completed traces :: 100%|##########| 150370/150370 [00:23<00:00, 6339.97it/s]","\n","[data] Loaded datasets: ['BPI2012', 'BPI2017', 'ROAD']","\n","\n=== Dataset: BPI2012 ===","\n","Samples train/val/test: 22252/4176/4196; vocab=23","\n","Epoch 1: validation_loss = 0.5594 | val_acc=0.7409 | val_f1=0.5316 | val_top3=0.9840","\n","Epoch 2: validation_loss = 0.5308 | val_acc=0.7598 | val_f1=0.5425 | val_top3=0.9854","\n","Epoch 3: validation_loss = 0.5333 | val_acc=0.7574 | val_f1=0.5725 | val_top3=0.9859","\n","Epoch 4: validation_loss = 0.5195 | val_acc=0.7629 | val_f1=0.5835 | val_top3=0.9861","\n","Epoch 5: validation_loss = 0.5221 | val_acc=0.7450 | val_f1=0.6007 | val_top3=0.9854","\n","Epoch 6: validation_loss = 0.5249 | val_acc=0.7634 | val_f1=0.5833 | val_top3=0.9861","\n","Epoch 7: validation_loss = 0.5073 | val_acc=0.7639 | val_f1=0.5909 | val_top3=0.9856","\n","Epoch 8: validation_loss = 0.5113 | val_acc=0.7593 | val_f1=0.5790 | val_top3=0.9859","\n","Epoch 9: validation_loss = 0.5105 | val_acc=0.7639 | val_f1=0.5842 | val_top3=0.9852","\n","Epoch 10: validation_loss = 0.5201 | val_acc=0.7634 | val_f1=0.5797 | val_top3=0.9847","\n","[BPI2012] Train: loss=0.5148 acc=0.7777 f1=0.5609 top3=0.9868","\n","[BPI2012] Test:  loss=0.5355 acc=0.7569 f1=0.5872 top3=0.9874","\n","\n=== Dataset: BPI2017 ===","\n","Samples train/val/test: 34519/7403/7374; vocab=24","\n","Epoch 1: validation_loss = 0.4172 | val_acc=0.8368 | val_f1=0.5299 | val_top3=0.9904","\n","Epoch 2: validation_loss = 0.4004 | val_acc=0.8367 | val_f1=0.5595 | val_top3=0.9907","\n","Epoch 3: validation_loss = 0.3864 | val_acc=0.8384 | val_f1=0.5719 | val_top3=0.9912","\n","Epoch 4: validation_loss = 0.3847 | val_acc=0.8395 | val_f1=0.5903 | val_top3=0.9919","\n","Epoch 5: validation_loss = 0.3812 | val_acc=0.8405 | val_f1=0.5856 | val_top3=0.9922","\n","Epoch 6: validation_loss = 0.3802 | val_acc=0.8376 | val_f1=0.5896 | val_top3=0.9916","\n","Epoch 7: validation_loss = 0.3769 | val_acc=0.8361 | val_f1=0.5957 | val_top3=0.9924","\n","Epoch 8: validation_loss = 0.3799 | val_acc=0.8386 | val_f1=0.6180 | val_top3=0.9927","\n","Epoch 9: validation_loss = 0.3805 | val_acc=0.8391 | val_f1=0.5989 | val_top3=0.9926","\n","Epoch 10: validation_loss = 0.3756 | val_acc=0.8399 | val_f1=0.6155 | val_top3=0.9928","\n","[BPI2017] Train: loss=0.3607 acc=0.8422 f1=0.5721 top3=0.9941","\n","[BPI2017] Test:  loss=0.3877 acc=0.8332 f1=0.5710 top3=0.9906","\n","\n=== Dataset: ROAD ===","\n","Samples train/val/test: 8707/1928/1869; vocab=11","\n","Epoch 1: validation_loss = 0.4913 | val_acc=0.7993 | val_f1=0.3684 | val_top3=0.9922","\n","Epoch 2: validation_loss = 0.4561 | val_acc=0.8086 | val_f1=0.4514 | val_top3=0.9948","\n","Epoch 3: validation_loss = 0.4566 | val_acc=0.8091 | val_f1=0.5300 | val_top3=0.9969","\n","Epoch 4: validation_loss = 0.4482 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 5: validation_loss = 0.4355 | val_acc=0.8112 | val_f1=0.6427 | val_top3=0.9969","\n","Epoch 6: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 7: validation_loss = 0.4328 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 8: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 9: validation_loss = 0.4274 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 10: validation_loss = 0.4329 | val_acc=0.8122 | val_f1=0.6664 | val_top3=0.9969","\n","[ROAD] Train: loss=0.4662 acc=0.7894 f1=0.5395 top3=0.9986","\n","[ROAD] Test:  loss=0.4833 acc=0.8020 f1=0.4740 top3=0.9936","\n","Execution time: 3 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"I will load the numpy file from the specified working directory, parse the nested experiment_data structure, and compute final or best metrics as requested. For training and test, I will report the final metrics saved in the structure. For validation, I will compute the best values across epochs for accuracy, F1 score, and top-3 accuracy, and the minimum validation loss using the recorded loss history. The script prints the dataset name followed by clearly labeled metrics and executes immediately.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef safe_get(d, *keys, default=None):\n    cur = d\n    for k in keys:\n        if isinstance(cur, dict) and k in cur:\n            cur = cur[k]\n        else:\n            return default\n    return cur\n\n\ndef load_experiment_data():\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    path = os.path.join(working_dir, \"experiment_data.npy\")\n    if not os.path.isfile(path):\n        raise FileNotFoundError(f\"File not found: {path}\")\n    data = np.load(path, allow_pickle=True).item()\n    if not isinstance(data, dict):\n        raise ValueError(\"Loaded experiment_data is not a dict.\")\n    return data\n\n\ndef extract_final_train_metrics(ds_data):\n    # Expect a 'final' entry\n    train_entries = safe_get(ds_data, \"metrics\", \"train\", default=[])\n    final = None\n    for tag, vals in train_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"train loss\"] = final.get(\"loss\", None)\n        out[\"train accuracy\"] = final.get(\"acc\", None)\n        out[\"train F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"train top-3 accuracy\"] = final.get(\"top3\", None)\n    else:\n        # Fallback: try last value from losses for loss only\n        train_losses = safe_get(ds_data, \"losses\", \"train\", default=[])\n        if train_losses:\n            out[\"train loss\"] = train_losses[-1][1]\n    return out\n\n\ndef extract_best_val_metrics(ds_data):\n    # Collect per-epoch validation metrics (acc, macro_f1, top3)\n    val_entries = safe_get(ds_data, \"metrics\", \"val\", default=[])\n    best = {\n        \"validation accuracy\": None,\n        \"validation F1 score\": None,\n        \"validation top-3 accuracy\": None,\n    }\n    for tag, vals in val_entries:\n        if isinstance(vals, dict):\n            acc = vals.get(\"acc\", None)\n            f1 = vals.get(\"macro_f1\", None)\n            top3 = vals.get(\"top3\", None)\n            if acc is not None:\n                best[\"validation accuracy\"] = (\n                    acc\n                    if best[\"validation accuracy\"] is None\n                    else max(best[\"validation accuracy\"], acc)\n                )\n            if f1 is not None:\n                best[\"validation F1 score\"] = (\n                    f1\n                    if best[\"validation F1 score\"] is None\n                    else max(best[\"validation F1 score\"], f1)\n                )\n            if top3 is not None:\n                best[\"validation top-3 accuracy\"] = (\n                    top3\n                    if best[\"validation top-3 accuracy\"] is None\n                    else max(best[\"validation top-3 accuracy\"], top3)\n                )\n    # For validation loss, use recorded losses per epoch and take the minimum\n    val_losses = safe_get(ds_data, \"losses\", \"val\", default=[])\n    if val_losses:\n        min_val_loss = min((v for (_, v) in val_losses), default=None)\n    else:\n        # fallback: check 'final' val entry if present\n        min_val_loss = None\n        for tag, vals in val_entries:\n            if tag == \"final\" and isinstance(vals, dict):\n                min_val_loss = vals.get(\"loss\", None)\n                break\n    out = {\"validation loss\": min_val_loss}\n    out.update(best)\n    return out\n\n\ndef extract_final_test_metrics(ds_data):\n    test_entries = safe_get(ds_data, \"metrics\", \"test\", default=[])\n    final = None\n    for tag, vals in test_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"test loss\"] = final.get(\"loss\", None)\n        out[\"test accuracy\"] = final.get(\"acc\", None)\n        out[\"test F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"test top-3 accuracy\"] = final.get(\"top3\", None)\n    return out\n\n\ndef format_print_metrics(dataset_name, metrics_dict):\n    print(dataset_name)\n    # Keep a consistent order for readability\n    ordered_keys = [\n        \"train loss\",\n        \"train accuracy\",\n        \"train F1 score\",\n        \"train top-3 accuracy\",\n        \"validation loss\",\n        \"validation accuracy\",\n        \"validation F1 score\",\n        \"validation top-3 accuracy\",\n        \"test loss\",\n        \"test accuracy\",\n        \"test F1 score\",\n        \"test top-3 accuracy\",\n    ]\n    printed_any = False\n    for k in ordered_keys:\n        if k in metrics_dict and metrics_dict[k] is not None:\n            val = metrics_dict[k]\n            if isinstance(val, float):\n                print(f\"{k}: {val:.4f}\")\n            else:\n                print(f\"{k}: {val}\")\n            printed_any = True\n    # Print any remaining keys that were not in the ordered list\n    for k, v in metrics_dict.items():\n        if k not in ordered_keys and v is not None:\n            if isinstance(v, float):\n                print(f\"{k}: {v:.4f}\")\n            else:\n                print(f\"{k}: {v}\")\n            printed_any = True\n    if not printed_any:\n        print(\"No metrics available\")\n\n\ndef run():\n    experiment_data = load_experiment_data()\n    for ds_name, ds_data in experiment_data.items():\n        all_metrics = {}\n        all_metrics.update(extract_final_train_metrics(ds_data))\n        all_metrics.update(extract_best_val_metrics(ds_data))\n        all_metrics.update(extract_final_test_metrics(ds_data))\n        format_print_metrics(ds_name, all_metrics)\n\n\n# Execute immediately\nrun()\n","parse_term_out":["BPI2012","\n","train loss: 0.5148","\n","train accuracy: 0.7777","\n","train F1 score: 0.5609","\n","train top-3 accuracy: 0.9868","\n","validation loss: 0.5073","\n","validation accuracy: 0.7639","\n","validation F1 score: 0.6007","\n","validation top-3 accuracy: 0.9861","\n","test loss: 0.5355","\n","test accuracy: 0.7569","\n","test F1 score: 0.5872","\n","test top-3 accuracy: 0.9874","\n","BPI2017","\n","train loss: 0.3607","\n","train accuracy: 0.8422","\n","train F1 score: 0.5721","\n","train top-3 accuracy: 0.9941","\n","validation loss: 0.3756","\n","validation accuracy: 0.8405","\n","validation F1 score: 0.6180","\n","validation top-3 accuracy: 0.9928","\n","test loss: 0.3877","\n","test accuracy: 0.8332","\n","test F1 score: 0.5710","\n","test top-3 accuracy: 0.9906","\n","ROAD","\n","train loss: 0.4662","\n","train accuracy: 0.7894","\n","train F1 score: 0.5395","\n","train top-3 accuracy: 0.9986","\n","validation loss: 0.4274","\n","validation accuracy: 0.8122","\n","validation F1 score: 0.6664","\n","validation top-3 accuracy: 0.9969","\n","test loss: 0.4833","\n","test accuracy: 0.8020","\n","test F1 score: 0.4740","\n","test top-3 accuracy: 0.9936","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":236.42285084724426,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"","exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087","metric":{"value":{"metric_names":[{"metric_name":"loss","lower_is_better":true,"description":"Cross-entropy loss averaged over samples.","data":[{"dataset_name":"BPI2012 tr","final_value":0.5148,"best_value":0.5148},{"dataset_name":"BPI2012 dev","final_value":0.5073,"best_value":0.5073},{"dataset_name":"BPI2012 ts","final_value":0.5355,"best_value":0.5355},{"dataset_name":"BPI2017 tr","final_value":0.3607,"best_value":0.3607},{"dataset_name":"BPI2017 dev","final_value":0.3756,"best_value":0.3756},{"dataset_name":"BPI2017 ts","final_value":0.3877,"best_value":0.3877},{"dataset_name":"ROAD tr","final_value":0.4662,"best_value":0.4662},{"dataset_name":"ROAD dev","final_value":0.4274,"best_value":0.4274},{"dataset_name":"ROAD ts","final_value":0.4833,"best_value":0.4833}]},{"metric_name":"accuracy","lower_is_better":false,"description":"Top-1 classification accuracy.","data":[{"dataset_name":"BPI2012 tr","final_value":0.7777,"best_value":0.7777},{"dataset_name":"BPI2012 dev","final_value":0.7639,"best_value":0.7639},{"dataset_name":"BPI2012 ts","final_value":0.7569,"best_value":0.7569},{"dataset_name":"BPI2017 tr","final_value":0.8422,"best_value":0.8422},{"dataset_name":"BPI2017 dev","final_value":0.8405,"best_value":0.8405},{"dataset_name":"BPI2017 ts","final_value":0.8332,"best_value":0.8332},{"dataset_name":"ROAD tr","final_value":0.7894,"best_value":0.7894},{"dataset_name":"ROAD dev","final_value":0.8122,"best_value":0.8122},{"dataset_name":"ROAD ts","final_value":0.802,"best_value":0.802}]},{"metric_name":"F1 score","lower_is_better":false,"description":"Harmonic mean of precision and recall (macro/weighted not specified).","data":[{"dataset_name":"BPI2012 tr","final_value":0.5609,"best_value":0.5609},{"dataset_name":"BPI2012 dev","final_value":0.6007,"best_value":0.6007},{"dataset_name":"BPI2012 ts","final_value":0.5872,"best_value":0.5872},{"dataset_name":"BPI2017 tr","final_value":0.5721,"best_value":0.5721},{"dataset_name":"BPI2017 dev","final_value":0.618,"best_value":0.618},{"dataset_name":"BPI2017 ts","final_value":0.571,"best_value":0.571},{"dataset_name":"ROAD tr","final_value":0.5395,"best_value":0.5395},{"dataset_name":"ROAD dev","final_value":0.6664,"best_value":0.6664},{"dataset_name":"ROAD ts","final_value":0.474,"best_value":0.474}]},{"metric_name":"top-3 accuracy","lower_is_better":false,"description":"Fraction of samples where the correct label is among the top-3 predictions.","data":[{"dataset_name":"BPI2012 tr","final_value":0.9868,"best_value":0.9868},{"dataset_name":"BPI2012 dev","final_value":0.9861,"best_value":0.9861},{"dataset_name":"BPI2012 ts","final_value":0.9874,"best_value":0.9874},{"dataset_name":"BPI2017 tr","final_value":0.9941,"best_value":0.9941},{"dataset_name":"BPI2017 dev","final_value":0.9928,"best_value":0.9928},{"dataset_name":"BPI2017 ts","final_value":0.9906,"best_value":0.9906},{"dataset_name":"ROAD tr","final_value":0.9986,"best_value":0.9986},{"dataset_name":"ROAD dev","final_value":0.9969,"best_value":0.9969},{"dataset_name":"ROAD ts","final_value":0.9936,"best_value":0.9936}]}]},"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":false,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":["../../logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_BPI2017.png","../../logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_BPI2012.png","../../logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_ROAD.png","../../logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_BPI2017.png","../../logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_ROAD.png","../../logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_BPI2012.png"],"plot_paths":["experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_BPI2012.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_ROAD.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_ROAD.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_BPI2012.png"],"plot_analyses":[{"analysis":"The validation top-3 accuracy for the BPI2017 dataset shows a consistent increase over the epochs, indicating that the model is effectively learning to predict the top-3 next activities over time. The accuracy surpasses 0.9925, which suggests strong predictive performance.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_BPI2017.png"},{"analysis":"The validation top-3 accuracy for the BPI2012 dataset starts strong, reaching above 0.9860 early in training, but shows a slight decline towards the end of the epochs. This could indicate overfitting or a plateau in learning efficiency, requiring potential adjustments in model complexity or training duration.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_BPI2012.png"},{"analysis":"The loss curves for the ROAD dataset show a clear and rapid decline in both training and validation loss. The validation loss stabilizes close to the training loss, indicating good generalization without significant overfitting. The model appears to be well-calibrated for this dataset.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_ROAD.png"},{"analysis":"The loss curves for the BPI2017 dataset demonstrate a sharp decline in both training and validation loss, with the validation loss closely following the training loss. This suggests effective learning and good model generalization with minimal overfitting.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_BPI2017.png"},{"analysis":"The validation top-3 accuracy for the ROAD dataset reaches a high value early on and remains stable throughout the epochs, indicating strong model performance and consistency in predicting the top-3 next activities. The accuracy remains nearly constant at around 0.997, reflecting excellent predictive capabilities.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/val_top3_ROAD.png"},{"analysis":"The loss curves for the BPI2012 dataset show a significant decrease in both training and validation loss with epochs, indicating effective learning. The validation loss is slightly higher than the training loss towards the end, suggesting minor overfitting that may be rectified with techniques such as dropout or regularization.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_726b2721d45c4800b9381c5d265fefcb_proc_332087/loss_curves_BPI2012.png"}],"vlm_feedback_summary":"The experimental results demonstrate strong predictive performance across different datasets, particularly in top-3 accuracy metrics. While the ROAD dataset shows excellent consistency and high accuracy, the BPI2012 dataset indicates potential overfitting concerns. Loss curves across datasets suggest good generalization capabilities, with minor overfitting observed in the BPI2012 dataset. Further refinement may be needed in model tuning and complexity to address these issues.","datasets_successfully_tested":["BPI2017","BPI2012","ROAD"],"ablation_name":null,"hyperparam_name":null,"is_seed_node":false,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"import os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nfrom datetime import datetime\nimport time\nimport math\n\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data container\nexperiment_data = {\n    \"BPI_base\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n        \"timestamp\": [],\n    }\n}\n\n# ------------- Data discovery and loading (pm4py XES) -------------\nfrom pathlib import Path\nfrom typing import Dict, List, Optional, Tuple\n\n\ndef _has_xes(dirpath: Path) -> bool:\n    try:\n        return dirpath.is_dir() and (\n            any(dirpath.glob(\"*.xes\")) or any(dirpath.glob(\"*.xes.gz\"))\n        )\n    except Exception:\n        return False\n\n\ndef _resolve_data_dir() -> Path:\n    candidates: List[Path] = []\n    candidates += [Path(\"input\").resolve(), (Path.cwd() / \"input\").resolve()]\n    cwd = Path.cwd().resolve()\n    for base in [cwd, *cwd.parents]:\n        candidates.append((base / \"data\").resolve())\n        candidates.append((base / \"input\").resolve())\n    candidates += [\n        Path(\"/workspace/input\"),\n        Path(\"/workspace/data\"),\n        Path(\"/workspace/ai_scientist/data\"),\n        Path(\"/workspace/AI-Scientist-v2/data\"),\n        Path(\"/workspace/experiments/data\"),\n        Path(\"/workspace/ai_scientist/input\"),\n        Path(\"/workspace/experiments/input\"),\n    ]\n    seen = set()\n    for p in candidates:\n        if p in seen:\n            continue\n        seen.add(p)\n        if _has_xes(p):\n            print(f\"[data] Using discovered data dir: {p}\")\n            return p\n    raise FileNotFoundError(\"No directory containing .xes or .xes.gz found.\")\n\n\ndef _first_match(d: Path, patterns: List[str]) -> Optional[Path]:\n    for pat in patterns:\n        for p in d.glob(pat):\n            if p.is_file():\n                return p\n    return None\n\n\ndef xes_to_df(xes_path: Path) -> pd.DataFrame:\n    try:\n        from pm4py.objects.log.importer.xes import importer as xes_importer\n    except Exception as e:\n        raise ImportError(\"pm4py is required. Install with `pip install pm4py`.\") from e\n    print(f\"[data] Loading XES: {xes_path}\")\n    log = xes_importer.apply(str(xes_path))\n    rows = []\n    for tr in log:\n        case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n            \"case:concept:name\"\n        )\n        for e in tr:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": e.get(\"concept:name\"),\n                    \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                    \"timestamp\": e.get(\"time:timestamp\"),\n                    \"resource\": e.get(\"org:resource\", \"System\"),\n                }\n            )\n    df = pd.DataFrame(rows)\n    df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n    df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n    df = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n    return df[[\"case_id\", \"activity\", \"lifecycle\", \"timestamp\", \"resource\"]]\n\n\ndef load_default_dataset() -> Tuple[str, pd.DataFrame]:\n    try:\n        d = _resolve_data_dir()\n        avail = sorted(\n            [p.name for p in list(d.glob(\"*.xes\")) + list(d.glob(\"*.xes.gz\"))]\n        )\n        print(f\"[data] Available: {avail}\")\n        patterns = {\n            \"BPI2017\": [\"BPI_Challenge_2017*.xes*\", \"BPI2017*.xes*\", \"*2017*.xes*\"],\n            \"BPI2012\": [\"BPI_Challenge_2012*.xes*\", \"BPI2012*.xes*\", \"*2012*.xes*\"],\n            \"ROAD\": [\n                \"Road_Traffic_Fine_Management_Process*.xes*\",\n                \"*Traffic*Fine*.xes*\",\n                \"*Traffic*.xes*\",\n            ],\n        }\n        for name in (\"BPI2012\", \"BPI2017\", \"ROAD\"):\n            path = _first_match(d, patterns[name])\n            if path is not None:\n                return name, xes_to_df(path)\n        # fallback any file\n        files = list(d.glob(\"*.xes\")) + list(d.glob(\"*.xes.gz\"))\n        if files:\n            return files[0].stem, xes_to_df(files[0])\n        raise FileNotFoundError(\"No XES files matched.\")\n    except Exception as e:\n        print(f\"[warn] XES load failed: {e}. Generating synthetic toy log.\")\n        # Synthetic tiny log: 3 activities, 50 cases\n        np.random.seed(42)\n        acts = [\"A\", \"B\", \"C\", \"D\", \"E\"]\n        rows = []\n        start = pd.Timestamp(\"2020-01-01\", tz=\"UTC\")\n        for cid in range(50):\n            length = np.random.randint(3, 7)\n            t = (\n                start\n                + pd.Timedelta(days=np.random.randint(0, 30))\n                + pd.Timedelta(minutes=np.random.randint(0, 1440))\n            )\n            case_id = f\"C{cid:04d}\"\n            seq = [\"A\"] + list(np.random.choice(acts[1:], size=length - 1))\n            for a in seq:\n                rows.append(\n                    {\n                        \"case_id\": case_id,\n                        \"activity\": a,\n                        \"lifecycle\": \"complete\",\n                        \"timestamp\": t,\n                        \"resource\": np.random.choice([\"R1\", \"R2\", \"R3\"]),\n                    }\n                )\n                t = t + pd.Timedelta(minutes=np.random.randint(1, 120))\n        df = (\n            pd.DataFrame(rows)\n            .sort_values([\"case_id\", \"timestamp\"])\n            .reset_index(drop=True)\n        )\n        return \"SYNTHETIC\", df\n\n\n# ------------- Prefix building and features -------------\ndef build_prefix_samples(df: pd.DataFrame, max_k: int = 10):\n    # Ensure per-case time order\n    df = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n    # Case start times for time-based split\n    case_start = (\n        df.groupby(\"case_id\")[\"timestamp\"]\n        .min()\n        .reset_index()\n        .rename(columns={\"timestamp\": \"case_start\"})\n    )\n    # Build sequences per case\n    samples = []\n    for cid, grp in df.groupby(\"case_id\"):\n        grp = grp.sort_values(\"timestamp\")\n        acts = grp[\"activity\"].tolist()\n        times = grp[\"timestamp\"].tolist()\n        res = grp[\"resource\"].tolist()\n        if len(acts) < 2:\n            continue\n        for k in range(1, min(len(acts), max_k + 1)):\n            prefix_acts = acts[:k]\n            prefix_times = times[:k]\n            target = acts[k] if k < len(acts) else None\n            if target is None:\n                continue\n            # per-step time features\n            t0 = prefix_times[0]\n            times_since_start = [\n                (t - t0).total_seconds() / 3600.0 for t in prefix_times\n            ]\n            times_since_prev = [0.0] + [\n                (prefix_times[i] - prefix_times[i - 1]).total_seconds() / 3600.0\n                for i in range(1, len(prefix_times))\n            ]\n            hours = [t.hour + t.minute / 60.0 for t in prefix_times]\n            weekdays = [t.weekday() for t in prefix_times]\n            working = [\n                1.0 if (wd < 5 and 8 <= h <= 18) else 0.0\n                for wd, h in zip(weekdays, hours)\n            ]\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"prefix_acts\": prefix_acts,\n                    \"times_since_start\": times_since_start,\n                    \"times_since_prev\": times_since_prev,\n                    \"hours\": hours,\n                    \"weekdays\": weekdays,\n                    \"working\": working,\n                    \"target\": target,\n                    \"case_start\": t0,\n                }\n            )\n    return pd.DataFrame(samples), case_start\n\n\n# ------------- Vocabulary and encoding -------------\nclass Vocab:\n    def __init__(self, tokens, add_unk=True, add_pad=True):\n        uniq = sorted(set(tokens))\n        self.pad_token = \"<PAD>\" if add_pad else None\n        self.unk_token = \"<UNK>\" if add_unk else None\n        idx = 0\n        self.token_to_id = {}\n        if add_pad:\n            self.token_to_id[self.pad_token] = idx\n            idx += 1\n        if add_unk:\n            self.token_to_id[self.unk_token] = idx\n            idx += 1\n        for t in uniq:\n            if t in (self.pad_token, self.unk_token):\n                continue\n            self.token_to_id[t] = idx\n            idx += 1\n        self.id_to_token = {v: k for k, v in self.token_to_id.items()}\n\n    def __len__(self):\n        return len(self.token_to_id)\n\n    def encode(self, t):\n        if t in self.token_to_id:\n            return self.token_to_id[t]\n        if self.unk_token is not None:\n            return self.token_to_id[self.unk_token]\n        raise KeyError(t)\n\n    def pad_id(self):\n        return self.token_to_id[self.pad_token] if self.pad_token else 0\n\n\n# ------------- Dataset and collate -------------\nclass PrefixDataset(Dataset):\n    def __init__(self, df, act_vocab: Vocab, target_vocab: Vocab, norm_stats: dict):\n        self.df = df\n        self.act_vocab = act_vocab\n        self.target_vocab = target_vocab\n        self.norm = norm_stats\n\n    def __len__(self):\n        return len(self.df)\n\n    def __getitem__(self, idx):\n        row = self.df.iloc[idx]\n        acts = torch.tensor(\n            [self.act_vocab.encode(a) for a in row[\"prefix_acts\"]], dtype=torch.long\n        )\n        # numeric features per step\n        feats = np.stack(\n            [\n                row[\"times_since_start\"],\n                row[\"times_since_prev\"],\n                row[\"hours\"],\n                row[\"weekdays\"],\n                row[\"working\"],\n            ],\n            axis=1,\n        ).astype(np.float32)\n        # normalize using train stats\n        mu = self.norm[\"mean\"]  # shape (5,)\n        sigma = self.norm[\"std\"]\n        feats = (feats - mu) / (sigma + 1e-8)\n        feats = torch.tensor(feats, dtype=torch.float32)\n        y = torch.tensor(self.target_vocab.encode(row[\"target\"]), dtype=torch.long)\n        return {\"acts\": acts, \"feats\": feats, \"y\": y}\n\n\ndef collate_batch(batch, pad_id, feat_dim):\n    lengths = [len(b[\"acts\"]) for b in batch]\n    max_len = max(lengths)\n    B = len(batch)\n    acts = torch.full((B, max_len), pad_id, dtype=torch.long)\n    feats = torch.zeros((B, max_len, feat_dim), dtype=torch.float32)\n    ys = torch.zeros((B,), dtype=torch.long)\n    for i, b in enumerate(batch):\n        L = len(b[\"acts\"])\n        acts[i, :L] = b[\"acts\"]\n        feats[i, :L] = b[\"feats\"]\n        ys[i] = b[\"y\"]\n    return {\n        \"acts\": acts,\n        \"feats\": feats,\n        \"y\": ys,\n        \"lengths\": torch.tensor(lengths, dtype=torch.long),\n    }\n\n\n# ------------- Model -------------\nclass LSTMNextAct(nn.Module):\n    def __init__(self, num_acts, emb_dim, feat_dim, hidden, num_classes, pad_idx):\n        super().__init__()\n        self.emb = nn.Embedding(num_acts, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + feat_dim, hidden_size=hidden, batch_first=True\n        )\n        self.drop = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, num_classes)\n\n    def forward(self, acts, feats, lengths):\n        x_emb = self.emb(acts)  # (B,T,E)\n        x = torch.cat([x_emb, feats], dim=-1)\n        # pack for efficiency\n        packed = nn.utils.rnn.pack_padded_sequence(\n            x, lengths.cpu(), batch_first=True, enforce_sorted=False\n        )\n        out_packed, (h, c) = self.lstm(packed)\n        h_last = h[-1]  # (B,H)\n        h_last = self.drop(h_last)\n        logits = self.fc(h_last)\n        return logits\n\n\n# ------------- Metrics -------------\ndef topk_accuracy(probs, y_true, k=3):\n    topk = probs.topk(k, dim=1).indices  # (B,k)\n    correct = (topk == y_true.unsqueeze(1)).any(dim=1).float()\n    return correct.mean().item()\n\n\ndef eval_model(model, loader, criterion):\n    model.eval()\n    all_logits = []\n    all_y = []\n    losses = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"lengths\"])\n            loss = criterion(logits, batch[\"y\"])\n            losses.append(loss.item())\n            all_logits.append(logits.cpu())\n            all_y.append(batch[\"y\"].cpu())\n    logits = torch.cat(all_logits, dim=0)\n    y = torch.cat(all_y, dim=0)\n    probs = torch.softmax(logits, dim=1)\n    y_pred = probs.argmax(dim=1).numpy()\n    y_true = y.numpy()\n    acc = accuracy_score(y_true, y_pred)\n    macro_f1 = f1_score(y_true, y_pred, average=\"macro\")\n    top3 = topk_accuracy(probs, y, k=3)\n    return np.mean(losses), acc, macro_f1, top3, probs.numpy(), y_true\n\n\n# ------------- Pipeline -------------\ndef run_experiment():\n    dataset_name, df = load_default_dataset()\n    print(\n        f\"[info] Using dataset: {dataset_name}, events={len(df)}, cases={df['case_id'].nunique()}\"\n    )\n    samples_df, case_start = build_prefix_samples(df, max_k=10)\n    if len(samples_df) == 0:\n        raise RuntimeError(\"No samples could be built from the event log.\")\n    # Time-based split by case start\n    case_times = samples_df.groupby(\"case_id\")[\"case_start\"].min().reset_index()\n    case_times = case_times.sort_values(\"case_start\")\n    n_cases = len(case_times)\n    n_train = int(0.7 * n_cases)\n    n_val = int(0.15 * n_cases)\n    train_cases = set(case_times.iloc[:n_train][\"case_id\"])\n    val_cases = set(case_times.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(case_times.iloc[n_train + n_val :][\"case_id\"])\n    train_df = samples_df[samples_df[\"case_id\"].isin(train_cases)].reset_index(\n        drop=True\n    )\n    val_df = samples_df[samples_df[\"case_id\"].isin(val_cases)].reset_index(drop=True)\n    test_df = samples_df[samples_df[\"case_id\"].isin(test_cases)].reset_index(drop=True)\n    print(\n        f\"[split] train={len(train_df)} val={len(val_df)} test={len(test_df)} samples\"\n    )\n\n    # Vocabularies (fit on TRAIN ONLY for targets; activity embeddings can include all seen to avoid UNK explosion)\n    act_vocab = Vocab(tokens=df[\"activity\"].tolist(), add_unk=True, add_pad=True)\n    target_vocab = Vocab(\n        tokens=train_df[\"target\"].tolist(), add_unk=True, add_pad=False\n    )\n\n    # Compute normalization stats on TRAIN numeric features (stack all steps)\n    def stack_numeric(df_):\n        arrs = []\n        for _, r in df_.iterrows():\n            feats = np.stack(\n                [\n                    r[\"times_since_start\"],\n                    r[\"times_since_prev\"],\n                    r[\"hours\"],\n                    r[\"weekdays\"],\n                    r[\"working\"],\n                ],\n                axis=1,\n            ).astype(np.float32)\n            arrs.append(feats)\n        if len(arrs) == 0:\n            return np.zeros((0, 5), dtype=np.float32)\n        return np.concatenate(arrs, axis=0)\n\n    train_feats_all = stack_numeric(train_df)\n    if train_feats_all.shape[0] == 0:\n        raise RuntimeError(\"Training features are empty.\")\n    mu = train_feats_all.mean(axis=0)\n    std = train_feats_all.std(axis=0)\n    std[std == 0] = 1.0\n    norm_stats = {\"mean\": mu, \"std\": std}\n\n    # Datasets and loaders\n    feat_dim = 5\n    pad_id = act_vocab.pad_id()\n    train_ds = PrefixDataset(train_df, act_vocab, target_vocab, norm_stats)\n    val_ds = PrefixDataset(val_df, act_vocab, target_vocab, norm_stats)\n    test_ds = PrefixDataset(test_df, act_vocab, target_vocab, norm_stats)\n    collate = lambda b: collate_batch(b, pad_id=pad_id, feat_dim=feat_dim)\n    train_loader = DataLoader(train_ds, batch_size=64, shuffle=True, collate_fn=collate)\n    val_loader = DataLoader(val_ds, batch_size=128, shuffle=False, collate_fn=collate)\n    test_loader = DataLoader(test_ds, batch_size=128, shuffle=False, collate_fn=collate)\n\n    # Model\n    num_acts = len(act_vocab)\n    num_classes = len(target_vocab)\n    model = LSTMNextAct(\n        num_acts=num_acts,\n        emb_dim=64,\n        feat_dim=feat_dim,\n        hidden=64,\n        num_classes=num_classes,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss()\n    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3)\n\n    # Training loop\n    epochs = 8\n    for epoch in range(1, epochs + 1):\n        model.train()\n        epoch_losses = []\n        for batch in train_loader:\n            batch = {\n                k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                for k, v in batch.items()\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"lengths\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            epoch_losses.append(loss.item())\n        train_loss = float(np.mean(epoch_losses)) if epoch_losses else float(\"nan\")\n        # Validation\n        val_loss, val_acc, val_f1, val_top3, _, _ = eval_model(\n            model, val_loader, criterion\n        )\n        print(\n            f\"Epoch {epoch}: train_loss = {train_loss:.4f} | validation_loss = {val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        # Record metrics\n        ts = time.time()\n        experiment_data[\"BPI_base\"][\"epochs\"].append(epoch)\n        experiment_data[\"BPI_base\"][\"timestamp\"].append(ts)\n        experiment_data[\"BPI_base\"][\"losses\"][\"train\"].append(\n            {\"epoch\": epoch, \"loss\": train_loss, \"t\": ts}\n        )\n        experiment_data[\"BPI_base\"][\"losses\"][\"val\"].append(\n            {\"epoch\": epoch, \"loss\": val_loss, \"t\": ts}\n        )\n        experiment_data[\"BPI_base\"][\"metrics\"][\"train\"].append(\n            {\"epoch\": epoch, \"metric\": \"loss\", \"value\": train_loss, \"t\": ts}\n        )\n        experiment_data[\"BPI_base\"][\"metrics\"][\"val\"].append(\n            {\n                \"epoch\": epoch,\n                \"accuracy\": val_acc,\n                \"macro_f1\": val_f1,\n                \"top3_accuracy\": val_top3,\n                \"validation_loss\": val_loss,\n                \"t\": ts,\n            }\n        )\n\n    # Final test evaluation\n    test_loss, test_acc, test_f1, test_top3, test_probs, test_y = eval_model(\n        model, test_loader, criterion\n    )\n    print(\n        f\"[test] loss={test_loss:.4f} acc={test_acc:.4f} macro_f1={test_f1:.4f} top3_acc={test_top3:.4f}\"\n    )\n\n    # Save predictions and ground truth (use class indices)\n    experiment_data[\"BPI_base\"][\"predictions\"] = test_probs\n    experiment_data[\"BPI_base\"][\"ground_truth\"] = test_y.tolist()\n    experiment_data[\"BPI_base\"][\"metrics\"][\"test\"].append(\n        {\n            \"loss\": test_loss,\n            \"accuracy\": test_acc,\n            \"macro_f1\": test_f1,\n            \"top3_accuracy\": test_top3,\n            \"t\": time.time(),\n            \"dataset\": dataset_name,\n        }\n    )\n\n    # Save all experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Also save compressed to be safe\n    np.savez_compressed(\n        os.path.join(working_dir, f\"next_act_results_{dataset_name}.npz\"),\n        experiment_data=experiment_data,\n    )\n\n\n# Execute\nrun_experiment()\n","plan":"We implement a minimal, reproducible PPM pipeline for next-activity prediction with a 1-layer LSTM baseline using pm4py-loaded local XES logs. We build prefix-based samples (k from 1 up to max_k=10) per case with process-aware temporal features: time since start, time since last event, hour-of-day, weekday, and a working-time flag. We enforce a strict time-based split at the case level (70/15/15 by case start time) to avoid leakage and normalize numeric features using training statistics only. The model embeds activities and concatenates per-step normalized temporal features, processes the sequence via an LSTM, and predicts the next activity at the last timestep. We train for a few epochs, track and print train/val losses and validation metrics each epoch, and finally evaluate on the held-out test set, reporting accuracy, macro-F1, and the primary metric: Top-3 Next-Activity Accuracy. Metrics, losses, predictions, and ground-truths are saved into experiment_data.npy, along with per-epoch timestamps. If no supported local XES logs are found, we synthesize a tiny toy log to complete the run. This simple baseline serves as a working foundation for subsequent resource-centric policy learning and simulation in later stages.","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import (\n    f1_score,\n    accuracy_score,\n    confusion_matrix,\n    precision_recall_curve,\n    auc,\n)\nfrom datetime import datetime\nimport time\nimport math\n\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data container\nexperiment_data = {\n    \"BPI_base\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n        \"timestamp\": [],\n    }\n}\n\n# -------- pm4py XES loading and utilities --------\nfrom pathlib import Path\nfrom typing import List, Optional, Tuple\n\n\ndef _has_xes(dirpath: Path) -> bool:\n    try:\n        return dirpath.is_dir() and (\n            any(dirpath.glob(\"*.xes\")) or any(dirpath.glob(\"*.xes.gz\"))\n        )\n    except Exception:\n        return False\n\n\ndef _resolve_data_dir() -> Path:\n    candidates: List[Path] = []\n    candidates += [Path(\"input\").resolve(), (Path.cwd() / \"input\").resolve()]\n    cwd = Path.cwd().resolve()\n    for base in [cwd, *cwd.parents]:\n        candidates.append((base / \"data\").resolve())\n        candidates.append((base / \"input\").resolve())\n    candidates += [\n        Path(\"/workspace/input\"),\n        Path(\"/workspace/data\"),\n        Path(\"/workspace/ai_scientist/data\"),\n        Path(\"/workspace/AI-Scientist-v2/data\"),\n        Path(\"/workspace/experiments/data\"),\n        Path(\"/workspace/ai_scientist/input\"),\n        Path(\"/workspace/experiments/input\"),\n    ]\n    seen = set()\n    for p in candidates:\n        if p in seen:\n            continue\n        seen.add(p)\n        if _has_xes(p):\n            print(f\"[data] Using discovered data dir: {p}\")\n            return p\n    raise FileNotFoundError(\"No directory containing .xes or .xes.gz found.\")\n\n\ndef _first_match(d: Path, patterns: List[str]) -> Optional[Path]:\n    for pat in patterns:\n        for p in d.glob(pat):\n            if p.is_file():\n                return p\n    return None\n\n\ndef xes_to_df(xes_path: Path) -> pd.DataFrame:\n    try:\n        from pm4py.objects.log.importer.xes import importer as xes_importer\n    except Exception as e:\n        raise ImportError(\"pm4py is required. Install with `pip install pm4py`.\") from e\n    print(f\"[data] Loading XES: {xes_path}\")\n    log = xes_importer.apply(str(xes_path))\n    rows = []\n    for tr in log:\n        case_id = tr.attributes.get(\"concept:name\") or tr.attributes.get(\n            \"case:concept:name\"\n        )\n        for e in tr:\n            rows.append(\n                {\n                    \"case_id\": case_id,\n                    \"activity\": e.get(\"concept:name\"),\n                    \"lifecycle\": e.get(\"lifecycle:transition\", \"complete\"),\n                    \"timestamp\": e.get(\"time:timestamp\"),\n                    \"resource\": e.get(\"org:resource\", \"System\"),\n                }\n            )\n    df = pd.DataFrame(rows)\n    df[\"timestamp\"] = pd.to_datetime(df[\"timestamp\"], utc=True, errors=\"coerce\")\n    df = df.dropna(subset=[\"timestamp\"]).reset_index(drop=True)\n    df = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n    return df[[\"case_id\", \"activity\", \"lifecycle\", \"timestamp\", \"resource\"]]\n\n\ndef load_default_dataset() -> Tuple[str, pd.DataFrame]:\n    try:\n        d = _resolve_data_dir()\n        avail = sorted(\n            [p.name for p in list(d.glob(\"*.xes\")) + list(d.glob(\"*.xes.gz\"))]\n        )\n        print(f\"[data] Available: {avail}\")\n        patterns = {\n            \"BPI2017\": [\"BPI_Challenge_2017*.xes*\", \"BPI2017*.xes*\", \"*2017*.xes*\"],\n            \"BPI2012\": [\"BPI_Challenge_2012*.xes*\", \"BPI2012*.xes*\", \"*2012*.xes*\"],\n            \"ROAD\": [\n                \"Road_Traffic_Fine_Management_Process*.xes*\",\n                \"*Traffic*Fine*.xes*\",\n                \"*Traffic*.xes*\",\n            ],\n        }\n        for name in (\"BPI2012\", \"BPI2017\", \"ROAD\"):\n            path = _first_match(d, patterns[name])\n            if path is not None:\n                return name, xes_to_df(path)\n        files = list(d.glob(\"*.xes\")) + list(d.glob(\"*.xes.gz\"))\n        if files:\n            return files[0].stem, xes_to_df(files[0])\n        raise FileNotFoundError(\"No XES files matched.\")\n    except Exception as e:\n        print(f\"[warn] XES load failed: {e}. Generating synthetic toy log.\")\n        np.random.seed(42)\n        acts = [\"A\", \"B\", \"C\", \"D\", \"E\"]\n        rows = []\n        start = pd.Timestamp(\"2020-01-01\", tz=\"UTC\")\n        for cid in range(50):\n            length = np.random.randint(3, 7)\n            t = (\n                start\n                + pd.Timedelta(days=np.random.randint(0, 30))\n                + pd.Timedelta(minutes=np.random.randint(0, 1440))\n            )\n            case_id = f\"C{cid:04d}\"\n            seq = [\"A\"] + list(np.random.choice(acts[1:], size=length - 1))\n            for a in seq:\n                rows.append(\n                    {\n                        \"case_id\": case_id,\n                        \"activity\": a,\n                        \"lifecycle\": \"complete\",\n                        \"timestamp\": t,\n                        \"resource\": np.random.choice([\"R1\", \"R2\", \"R3\"]),\n                    }\n                )\n                t = t + pd.Timedelta(minutes=np.random.randint(1, 120))\n        df = (\n            pd.DataFrame(rows)\n            .sort_values([\"case_id\", \"timestamp\"])\n            .reset_index(drop=True)\n        )\n        return \"SYNTHETIC\", df\n\n\n# -------- Prefix building --------\ndef build_prefix_samples(df: pd.DataFrame, max_k: int = 10):\n    df = df.sort_values([\"case_id\", \"timestamp\"]).reset_index(drop=True)\n    samples = []\n    for cid, grp in df.groupby(\"case_id\"):\n        grp = grp.sort_values(\"timestamp\")\n        acts = grp[\"activity\"].tolist()\n        times = grp[\"timestamp\"].tolist()\n        res = grp[\"resource\"].tolist()\n        if len(acts) < 2:\n            continue\n        for k in range(1, min(len(acts), max_k + 1)):\n            prefix_acts = acts[:k]\n            prefix_times = times[:k]\n            target = acts[k] if k < len(acts) else None\n            if target is None:\n                continue\n            t0 = prefix_times[0]\n            times_since_start = [\n                (t - t0).total_seconds() / 3600.0 for t in prefix_times\n            ]\n            times_since_prev = [0.0] + [\n                (prefix_times[i] - prefix_times[i - 1]).total_seconds() / 3600.0\n                for i in range(1, len(prefix_times))\n            ]\n            hours = [t.hour + t.minute / 60.0 for t in prefix_times]\n            weekdays = [t.weekday() for t in prefix_times]\n            working = [\n                1.0 if (wd < 5 and 8 <= h <= 18) else 0.0\n                for wd, h in zip(weekdays, hours)\n            ]\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"prefix_acts\": prefix_acts,\n                    \"times_since_start\": times_since_start,\n                    \"times_since_prev\": times_since_prev,\n                    \"hours\": hours,\n                    \"weekdays\": weekdays,\n                    \"working\": working,\n                    \"target\": target,\n                    \"case_start\": t0,\n                }\n            )\n    return pd.DataFrame(samples)\n\n\n# -------- Vocab --------\nclass Vocab:\n    def __init__(self, tokens, add_unk=True, add_pad=True):\n        uniq = sorted(set(tokens))\n        self.pad_token = \"<PAD>\" if add_pad else None\n        self.unk_token = \"<UNK>\" if add_unk else None\n        idx = 0\n        self.token_to_id = {}\n        if add_pad:\n            self.token_to_id[self.pad_token] = idx\n            idx += 1\n        if add_unk:\n            self.token_to_id[self.unk_token] = idx\n            idx += 1\n        for t in uniq:\n            if t in (self.pad_token, self.unk_token):\n                continue\n            self.token_to_id[t] = idx\n            idx += 1\n        self.id_to_token = {v: k for k, v in self.token_to_id.items()}\n\n    def __len__(self):\n        return len(self.token_to_id)\n\n    def encode(self, t):\n        if t in self.token_to_id:\n            return self.token_to_id[t]\n        if self.unk_token is not None:\n            return self.token_to_id[self.unk_token]\n        raise KeyError(t)\n\n    def pad_id(self):\n        return self.token_to_id[self.pad_token] if self.pad_token else 0\n\n\n# -------- Dataset --------\nclass PrefixDataset(Dataset):\n    def __init__(self, df, act_vocab: Vocab, target_vocab: Vocab, norm_stats: dict):\n        self.df = df\n        self.act_vocab = act_vocab\n        self.target_vocab = target_vocab\n        self.norm = norm_stats\n\n    def __len__(self):\n        return len(self.df)\n\n    def __getitem__(self, idx):\n        r = self.df.iloc[idx]\n        acts = torch.tensor(\n            [self.act_vocab.encode(a) for a in r[\"prefix_acts\"]], dtype=torch.long\n        )\n        feats = np.stack(\n            [\n                r[\"times_since_start\"],\n                r[\"times_since_prev\"],\n                r[\"hours\"],\n                r[\"weekdays\"],\n                r[\"working\"],\n            ],\n            axis=1,\n        ).astype(np.float32)\n        mu, sigma = self.norm[\"mean\"], self.norm[\"std\"]\n        feats = (feats - mu) / (sigma + 1e-8)\n        feats = torch.tensor(feats, dtype=torch.float32)\n        y = torch.tensor(self.target_vocab.encode(r[\"target\"]), dtype=torch.long)\n        return {\"acts\": acts, \"feats\": feats, \"y\": y}\n\n\ndef collate_batch(batch, pad_id, feat_dim):\n    lengths = [len(b[\"acts\"]) for b in batch]\n    max_len, B = max(lengths), len(batch)\n    acts = torch.full((B, max_len), pad_id, dtype=torch.long)\n    feats = torch.zeros((B, max_len, feat_dim), dtype=torch.float32)\n    ys = torch.zeros((B,), dtype=torch.long)\n    for i, b in enumerate(batch):\n        L = len(b[\"acts\"])\n        acts[i, :L] = b[\"acts\"]\n        feats[i, :L] = b[\"feats\"]\n        ys[i] = b[\"y\"]\n    return {\n        \"acts\": acts,\n        \"feats\": feats,\n        \"y\": ys,\n        \"lengths\": torch.tensor(lengths, dtype=torch.long),\n    }\n\n\n# -------- Model --------\nclass LSTMNextAct(nn.Module):\n    def __init__(self, num_acts, emb_dim, feat_dim, hidden, num_classes, pad_idx):\n        super().__init__()\n        self.emb = nn.Embedding(num_acts, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + feat_dim, hidden_size=hidden, batch_first=True\n        )\n        self.drop = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, num_classes)\n\n    def forward(self, acts, feats, lengths):\n        x = torch.cat([self.emb(acts), feats], dim=-1)\n        packed = nn.utils.rnn.pack_padded_sequence(\n            x, lengths.cpu(), batch_first=True, enforce_sorted=False\n        )\n        _, (h, _) = self.lstm(packed)\n        h = self.drop(h[-1])\n        return self.fc(h)\n\n\n# -------- Metrics --------\ndef topk_accuracy(probs, y_true, k=3):\n    topk = probs.topk(k, dim=1).indices\n    return (topk == y_true.unsqueeze(1)).any(dim=1).float().mean().item()\n\n\ndef eval_model(model, loader, criterion):\n    model.eval()\n    all_logits = []\n    all_y = []\n    losses = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"lengths\"])\n            loss = criterion(logits, batch[\"y\"])\n            losses.append(loss.item())\n            all_logits.append(logits.cpu())\n            all_y.append(batch[\"y\"].cpu())\n    logits = torch.cat(all_logits, 0)\n    y = torch.cat(all_y, 0)\n    probs = torch.softmax(logits, dim=1)\n    y_pred = probs.argmax(1).numpy()\n    y_true = y.numpy()\n    acc = accuracy_score(y_true, y_pred)\n    macro_f1 = f1_score(y_true, y_pred, average=\"macro\")\n    top3 = topk_accuracy(probs, y, k=3)\n    return np.mean(losses), acc, macro_f1, top3, probs.numpy(), y_true\n\n\n# -------- Run experiment --------\ndef run_experiment():\n    dataset_name, df = load_default_dataset()\n    print(\n        f\"[info] Using dataset: {dataset_name}, events={len(df)}, cases={df['case_id'].nunique()}\"\n    )\n    samples_df = build_prefix_samples(df, max_k=10)\n    if len(samples_df) == 0:\n        raise RuntimeError(\"No samples could be built from the event log.\")\n    case_times = (\n        samples_df.groupby(\"case_id\")[\"case_start\"]\n        .min()\n        .reset_index()\n        .sort_values(\"case_start\")\n    )\n    n_cases = len(case_times)\n    n_train = int(0.7 * n_cases)\n    n_val = int(0.15 * n_cases)\n    train_cases = set(case_times.iloc[:n_train][\"case_id\"])\n    val_cases = set(case_times.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(case_times.iloc[n_train + n_val :][\"case_id\"])\n    train_df = samples_df[samples_df[\"case_id\"].isin(train_cases)].reset_index(\n        drop=True\n    )\n    val_df = samples_df[samples_df[\"case_id\"].isin(val_cases)].reset_index(drop=True)\n    test_df = samples_df[samples_df[\"case_id\"].isin(test_cases)].reset_index(drop=True)\n    print(\n        f\"[split] train={len(train_df)} val={len(val_df)} test={len(test_df)} samples\"\n    )\n\n    act_vocab = Vocab(tokens=df[\"activity\"].tolist(), add_unk=True, add_pad=True)\n    target_vocab = Vocab(\n        tokens=train_df[\"target\"].tolist(), add_unk=True, add_pad=False\n    )\n\n    def stack_numeric(df_):\n        arrs = []\n        for _, r in df_.iterrows():\n            feats = np.stack(\n                [\n                    r[\"times_since_start\"],\n                    r[\"times_since_prev\"],\n                    r[\"hours\"],\n                    r[\"weekdays\"],\n                    r[\"working\"],\n                ],\n                axis=1,\n            ).astype(np.float32)\n            arrs.append(feats)\n        return np.concatenate(arrs, axis=0) if arrs else np.zeros((0, 5), np.float32)\n\n    train_feats_all = stack_numeric(train_df)\n    if train_feats_all.shape[0] == 0:\n        raise RuntimeError(\"Training features are empty.\")\n    mu = train_feats_all.mean(axis=0)\n    std = train_feats_all.std(axis=0)\n    std[std == 0] = 1.0\n    norm_stats = {\"mean\": mu, \"std\": std}\n\n    feat_dim = 5\n    pad_id = act_vocab.pad_id()\n    train_ds = PrefixDataset(train_df, act_vocab, target_vocab, norm_stats)\n    val_ds = PrefixDataset(val_df, act_vocab, target_vocab, norm_stats)\n    test_ds = PrefixDataset(test_df, act_vocab, target_vocab, norm_stats)\n    collate = lambda b: collate_batch(b, pad_id=pad_id, feat_dim=feat_dim)\n    train_loader = DataLoader(train_ds, batch_size=64, shuffle=True, collate_fn=collate)\n    val_loader = DataLoader(val_ds, batch_size=128, shuffle=False, collate_fn=collate)\n    test_loader = DataLoader(test_ds, batch_size=128, shuffle=False, collate_fn=collate)\n\n    num_acts = len(act_vocab)\n    num_classes = len(target_vocab)\n    model = LSTMNextAct(\n        num_acts=num_acts,\n        emb_dim=64,\n        feat_dim=feat_dim,\n        hidden=64,\n        num_classes=num_classes,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss()\n    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3)\n\n    epochs = 8\n    for epoch in range(1, epochs + 1):\n        model.train()\n        ep_losses = []\n        for batch in train_loader:\n            batch = {\n                k: (v.to(device) if isinstance(v, torch.Tensor) else v)\n                for k, v in batch.items()\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"lengths\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            ep_losses.append(loss.item())\n        train_loss = float(np.mean(ep_losses)) if ep_losses else float(\"nan\")\n        val_loss, val_acc, val_f1, val_top3, _, _ = eval_model(\n            model, val_loader, criterion\n        )\n        print(\n            f\"Epoch {epoch}: train_loss={train_loss:.4f} | val_loss={val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        ts = time.time()\n        experiment_data[\"BPI_base\"][\"epochs\"].append(epoch)\n        experiment_data[\"BPI_base\"][\"timestamp\"].append(ts)\n        experiment_data[\"BPI_base\"][\"losses\"][\"train\"].append(\n            {\"epoch\": epoch, \"loss\": train_loss, \"t\": ts}\n        )\n        experiment_data[\"BPI_base\"][\"losses\"][\"val\"].append(\n            {\"epoch\": epoch, \"loss\": val_loss, \"t\": ts}\n        )\n        experiment_data[\"BPI_base\"][\"metrics\"][\"train\"].append(\n            {\"epoch\": epoch, \"metric\": \"loss\", \"value\": train_loss, \"t\": ts}\n        )\n        experiment_data[\"BPI_base\"][\"metrics\"][\"val\"].append(\n            {\n                \"epoch\": epoch,\n                \"accuracy\": val_acc,\n                \"macro_f1\": val_f1,\n                \"top3_accuracy\": val_top3,\n                \"validation_loss\": val_loss,\n                \"t\": ts,\n            }\n        )\n\n    test_loss, test_acc, test_f1, test_top3, test_probs, test_y = eval_model(\n        model, test_loader, criterion\n    )\n    print(\n        f\"[test] loss={test_loss:.4f} acc={test_acc:.4f} macro_f1={test_f1:.4f} top3_acc={test_top3:.4f}\"\n    )\n    experiment_data[\"BPI_base\"][\"predictions\"] = test_probs\n    experiment_data[\"BPI_base\"][\"ground_truth\"] = test_y.tolist()\n    experiment_data[\"BPI_base\"][\"metrics\"][\"test\"].append(\n        {\n            \"loss\": test_loss,\n            \"accuracy\": test_acc,\n            \"macro_f1\": test_f1,\n            \"top3_accuracy\": test_top3,\n            \"t\": time.time(),\n            \"dataset\": dataset_name,\n        }\n    )\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    np.savez_compressed(\n        os.path.join(working_dir, f\"next_act_results_{dataset_name}.npz\"),\n        experiment_data=experiment_data,\n    )\n    return dataset_name\n\n\ndataset_name = run_experiment()\n\n# -------- Plotting from saved experiment_data.npy --------\ntry:\n    experiment_data = np.load(\n        os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n    ).item()\nexcept Exception as e:\n    print(f\"Error loading experiment data: {e}\")\n\nds_key = \"BPI_base\"\nds_name = None\ntry:\n    tests = experiment_data[ds_key][\"metrics\"][\"test\"]\n    ds_name = tests[-1].get(\"dataset\", \"UNKNOWN\") if tests else \"UNKNOWN\"\nexcept Exception:\n    ds_name = \"UNKNOWN\"\n\n# Training/Validation loss curves\ntry:\n    plt.figure()\n    train_losses = experiment_data[ds_key][\"losses\"][\"train\"]\n    val_losses = experiment_data[ds_key][\"losses\"][\"val\"]\n    if train_losses:\n        plt.plot(\n            [x[\"epoch\"] for x in train_losses],\n            [x[\"loss\"] for x in train_losses],\n            label=\"Train Loss\",\n        )\n    if val_losses:\n        plt.plot(\n            [x[\"epoch\"] for x in val_losses],\n            [x[\"loss\"] for x in val_losses],\n            label=\"Val Loss\",\n        )\n    plt.xlabel(\"Epoch\")\n    plt.ylabel(\"Loss\")\n    plt.title(\n        f\"{ds_name} Next-Activity: Training/Validation Loss\\nSubtitle: Standard curves from saved experiment_data.npy\"\n    )\n    plt.legend()\n    plt.savefig(\n        os.path.join(working_dir, f\"{ds_name}_next_activity_train_val_loss.png\")\n    )\n    plt.close()\nexcept Exception as e:\n    print(f\"Error creating loss curves: {e}\")\n    plt.close()\n\n# Validation metrics over epochs\ntry:\n    plt.figure()\n    val_metrics = experiment_data[ds_key][\"metrics\"][\"val\"]\n    if val_metrics:\n        epochs = [m[\"epoch\"] for m in val_metrics]\n        accs = [m.get(\"accuracy\", np.nan) for m in val_metrics]\n        f1s = [m.get(\"macro_f1\", np.nan) for m in val_metrics]\n        t3s = [m.get(\"top3_accuracy\", np.nan) for m in val_metrics]\n        plt.plot(epochs, accs, label=\"Val Accuracy\")\n        plt.plot(epochs, f1s, label=\"Val Macro-F1\")\n        plt.plot(epochs, t3s, label=\"Val Top-3 Acc\")\n    plt.xlabel(\"Epoch\")\n    plt.ylabel(\"Score\")\n    plt.title(\n        f\"{ds_name} Next-Activity: Validation Metrics\\nSubtitle: Accuracy, Macro-F1, Top-3\"\n    )\n    plt.legend()\n    plt.savefig(os.path.join(working_dir, f\"{ds_name}_next_activity_val_metrics.png\"))\n    plt.close()\nexcept Exception as e:\n    print(f\"Error creating val metrics plot: {e}\")\n    plt.close()\n\n# Confusion matrix (test)\ntry:\n    plt.figure()\n    test_probs = np.array(experiment_data[ds_key][\"predictions\"])\n    y_true = np.array(experiment_data[ds_key][\"ground_truth\"])\n    if test_probs.size > 0 and y_true.size > 0:\n        y_pred = test_probs.argmax(axis=1)\n        cm = confusion_matrix(y_true, y_pred)\n        im = plt.imshow(cm, cmap=\"Blues\")\n        plt.colorbar(im, fraction=0.046, pad=0.04)\n        num_classes = cm.shape[0]\n        plt.xlabel(\"Predicted\")\n        plt.ylabel(\"True\")\n        plt.title(f\"{ds_name} Next-Activity: Confusion Matrix\\nSubtitle: Test set\")\n        plt.savefig(\n            os.path.join(working_dir, f\"{ds_name}_next_activity_confusion_matrix.png\")\n        )\n    plt.close()\nexcept Exception as e:\n    print(f\"Error creating confusion matrix: {e}\")\n    plt.close()\n\n# Precision-Recall curves (micro and macro)\ntry:\n    plt.figure()\n    test_probs = np.array(experiment_data[ds_key][\"predictions\"])\n    y_true = np.array(experiment_data[ds_key][\"ground_truth\"])\n    if test_probs.size > 0 and y_true.size > 0:\n        n_classes = test_probs.shape[1]\n        # One-vs-rest PR for macro; micro with ravel\n        # Micro-average\n        y_true_ovr = np.eye(n_classes)[y_true]\n        precision_micro, recall_micro, _ = precision_recall_curve(\n            y_true_ovr.ravel(), test_probs.ravel()\n        )\n        ap_micro = auc(recall_micro, precision_micro)\n        plt.plot(recall_micro, precision_micro, label=f\"Micro-avg (AP={ap_micro:.3f})\")\n        # Macro-average (interpolate)\n        recalls = np.linspace(0, 1, 101)\n        precs = []\n        for c in range(n_classes):\n            p, r, _ = precision_recall_curve(y_true_ovr[:, c], test_probs[:, c])\n            precs.append(np.interp(recalls, r[::-1], p[::-1], left=p[0], right=p[-1]))\n        precision_macro = np.mean(precs, axis=0)\n        ap_macro = auc(recalls, precision_macro)\n        plt.plot(recalls, precision_macro, label=f\"Macro-avg (AP={ap_macro:.3f})\")\n        plt.xlabel(\"Recall\")\n        plt.ylabel(\"Precision\")\n        plt.title(\n            f\"{ds_name} Next-Activity: Precision-Recall Curves\\nSubtitle: Micro and Macro averages on Test\"\n        )\n        plt.legend()\n        plt.savefig(os.path.join(working_dir, f\"{ds_name}_next_activity_pr_curves.png\"))\n    plt.close()\nexcept Exception as e:\n    print(f\"Error creating PR curves: {e}\")\n    plt.close()\n\n# Print final BPM metrics\ntry:\n    test_entry = experiment_data[ds_key][\"metrics\"][\"test\"][-1]\n    print(\n        f\"Final Test Metrics [{ds_name}] -> loss={test_entry['loss']:.4f}, acc={test_entry['accuracy']:.4f}, macro_f1={test_entry['macro_f1']:.4f}, top3_acc={test_entry['top3_accuracy']:.4f}\"\n    )\nexcept Exception as e:\n    print(f\"Error printing final metrics: {e}\")\n","plot_plan":"We will implement a next-activity prediction pipeline using pm4py to load a local XES log (with a synthetic fallback), build time-ordered prefixes, and perform a strictly time-based case split (70/15/15) for train/val/test. An LSTM with activity embeddings plus time/context features will be trained, and we will record BPM metrics: accuracy, macro-F1, and Top-3 accuracy on validation and test sets. The code saves experiment_data.npy containing metrics, losses, predictions (probabilities), and ground-truth labels. After training, we load experiment_data.npy and create standard visualizations based solely on the saved arrays. Specifically, we plot training/validation loss curves and validation metric curves over epochs. For the next-activity task, we add a test confusion matrix and precision-recall curves (micro and macro averages) using the saved test probabilities and labels. We include careful try-except blocks per figure, save all plots to the working directory with descriptive names including the dataset, and close figures after saving. The script prints key evaluation metrics and avoids fabricating any missing data (e.g., skips Top-3 vs prefix-length plot because prefix lengths are not stored).","step":5,"id":"da7f619dc6be4f5fa90081d392370f2d","ctime":1757756700.2891667,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 8/13087 [00:00<02:43, 79.89it/s]","\rparsing log, completed traces ::   1%|1         | 173/13087 [00:00<00:12, 994.48it/s]","\rparsing log, completed traces ::   3%|2         | 333/13087 [00:00<00:10, 1265.09it/s]","\rparsing log, completed traces ::   4%|3         | 502/13087 [00:00<00:08, 1431.68it/s]","\rparsing log, completed traces ::   5%|5         | 677/13087 [00:00<00:08, 1542.84it/s]","\rparsing log, completed traces ::   7%|6         | 858/13087 [00:00<00:07, 1628.49it/s]","\rparsing log, completed traces ::   8%|7         | 1033/13087 [00:00<00:07, 1664.70it/s]","\rparsing log, completed traces ::   9%|9         | 1217/13087 [00:00<00:06, 1712.81it/s]","\rparsing log, completed traces ::  11%|#         | 1389/13087 [00:01<00:13, 847.17it/s] ","\rparsing log, completed traces ::  12%|#2        | 1603/13087 [00:01<00:10, 1076.63it/s]","\rparsing log, completed traces ::  14%|#3        | 1785/13087 [00:01<00:09, 1225.84it/s]","\rparsing log, completed traces ::  15%|#4        | 1948/13087 [00:01<00:08, 1307.87it/s]","\rparsing log, completed traces ::  16%|#6        | 2119/13087 [00:01<00:07, 1402.07it/s]","\rparsing log, completed traces ::  18%|#7        | 2293/13087 [00:01<00:07, 1488.10it/s]","\rparsing log, completed traces ::  19%|#9        | 2495/13087 [00:01<00:06, 1630.16it/s]","\rparsing log, completed traces ::  20%|##        | 2673/13087 [00:01<00:06, 1662.16it/s]","\rparsing log, completed traces ::  22%|##1       | 2850/13087 [00:02<00:06, 1616.28it/s]","\rparsing log, completed traces ::  23%|##3       | 3030/13087 [00:02<00:06, 1664.81it/s]","\rparsing log, completed traces ::  24%|##4       | 3202/13087 [00:02<00:05, 1652.68it/s]","\rparsing log, completed traces ::  26%|##6       | 3464/13087 [00:02<00:04, 1924.90it/s]","\rparsing log, completed traces ::  28%|##7       | 3661/13087 [00:02<00:05, 1879.88it/s]","\rparsing log, completed traces ::  29%|##9       | 3852/13087 [00:02<00:04, 1858.62it/s]","\rparsing log, completed traces ::  31%|###1      | 4057/13087 [00:02<00:08, 1016.51it/s]","\rparsing log, completed traces ::  33%|###3      | 4338/13087 [00:03<00:06, 1331.57it/s]","\rparsing log, completed traces ::  35%|###4      | 4567/13087 [00:03<00:05, 1524.74it/s]","\rparsing log, completed traces ::  36%|###6      | 4765/13087 [00:03<00:05, 1611.74it/s]","\rparsing log, completed traces ::  38%|###7      | 4969/13087 [00:03<00:04, 1712.89it/s]","\rparsing log, completed traces ::  39%|###9      | 5167/13087 [00:03<00:04, 1748.83it/s]","\rparsing log, completed traces ::  41%|####      | 5361/13087 [00:03<00:04, 1767.36it/s]","\rparsing log, completed traces ::  42%|####2     | 5552/13087 [00:03<00:04, 1704.97it/s]","\rparsing log, completed traces ::  44%|####3     | 5742/13087 [00:03<00:04, 1754.68it/s]","\rparsing log, completed traces ::  45%|####5     | 5933/13087 [00:03<00:03, 1797.53it/s]","\rparsing log, completed traces ::  47%|####6     | 6119/13087 [00:04<00:04, 1734.99it/s]","\rparsing log, completed traces ::  48%|####8     | 6335/13087 [00:04<00:03, 1850.60it/s]","\rparsing log, completed traces ::  50%|####9     | 6529/13087 [00:04<00:03, 1871.88it/s]","\rparsing log, completed traces ::  51%|#####1    | 6719/13087 [00:04<00:03, 1793.28it/s]","\rparsing log, completed traces ::  53%|#####2    | 6901/13087 [00:04<00:03, 1724.69it/s]","\rparsing log, completed traces ::  54%|#####4    | 7078/13087 [00:04<00:03, 1734.06it/s]","\rparsing log, completed traces ::  55%|#####5    | 7263/13087 [00:04<00:03, 1758.13it/s]","\rparsing log, completed traces ::  57%|#####6    | 7440/13087 [00:05<00:06, 874.45it/s] ","\rparsing log, completed traces ::  58%|#####8    | 7599/13087 [00:05<00:05, 994.63it/s]","\rparsing log, completed traces ::  59%|#####9    | 7763/13087 [00:05<00:04, 1119.51it/s]","\rparsing log, completed traces ::  61%|######    | 7922/13087 [00:05<00:04, 1220.94it/s]","\rparsing log, completed traces ::  62%|######1   | 8104/13087 [00:05<00:03, 1361.34it/s]","\rparsing log, completed traces ::  63%|######3   | 8288/13087 [00:05<00:03, 1482.04it/s]","\rparsing log, completed traces ::  65%|######4   | 8473/13087 [00:05<00:02, 1575.39it/s]","\rparsing log, completed traces ::  66%|######6   | 8655/13087 [00:05<00:02, 1641.80it/s]","\rparsing log, completed traces ::  68%|######7   | 8839/13087 [00:05<00:02, 1694.35it/s]","\rparsing log, completed traces ::  69%|######8   | 9017/13087 [00:06<00:02, 1683.00it/s]","\rparsing log, completed traces ::  70%|#######   | 9191/13087 [00:06<00:02, 1698.74it/s]","\rparsing log, completed traces ::  72%|#######1  | 9389/13087 [00:06<00:02, 1776.42it/s]","\rparsing log, completed traces ::  73%|#######3  | 9570/13087 [00:06<00:02, 1682.72it/s]","\rparsing log, completed traces ::  75%|#######4  | 9752/13087 [00:06<00:01, 1718.87it/s]","\rparsing log, completed traces ::  76%|#######5  | 9940/13087 [00:06<00:01, 1760.38it/s]","\rparsing log, completed traces ::  77%|#######7  | 10118/13087 [00:06<00:01, 1740.19it/s]","\rparsing log, completed traces ::  79%|#######8  | 10294/13087 [00:06<00:01, 1682.86it/s]","\rparsing log, completed traces ::  80%|#######9  | 10464/13087 [00:06<00:01, 1653.47it/s]","\rparsing log, completed traces ::  81%|########1 | 10631/13087 [00:07<00:01, 1634.93it/s]","\rparsing log, completed traces ::  82%|########2 | 10796/13087 [00:07<00:02, 768.73it/s] ","\rparsing log, completed traces ::  84%|########4 | 10997/13087 [00:07<00:02, 968.73it/s]","\rparsing log, completed traces ::  85%|########5 | 11181/13087 [00:07<00:01, 1126.22it/s]","\rparsing log, completed traces ::  87%|########6 | 11372/13087 [00:07<00:01, 1290.80it/s]","\rparsing log, completed traces ::  88%|########8 | 11567/13087 [00:07<00:01, 1441.73it/s]","\rparsing log, completed traces ::  90%|########9 | 11775/13087 [00:08<00:00, 1599.49it/s]","\rparsing log, completed traces ::  92%|#########1| 11993/13087 [00:08<00:00, 1746.69it/s]","\rparsing log, completed traces ::  93%|#########3| 12188/13087 [00:08<00:00, 1797.67it/s]","\rparsing log, completed traces ::  95%|#########4| 12402/13087 [00:08<00:00, 1889.28it/s]","\rparsing log, completed traces ::  96%|#########6| 12619/13087 [00:08<00:00, 1966.75it/s]","\rparsing log, completed traces ::  98%|#########8| 12826/13087 [00:08<00:00, 1995.40it/s]","\rparsing log, completed traces :: 100%|#########9| 13032/13087 [00:08<00:00, 2005.94it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:08<00:00, 1516.49it/s]","\n","[info] Using dataset: BPI2012, events=262200, cases=13087","\n","[split] train=63733 val=13924 test=13528 samples","\n","Epoch 1: train_loss = 0.5640 | validation_loss = 0.4001 | val_acc=0.8035 | val_f1=0.6793 | val_top3=0.9947","\n","Epoch 2: train_loss = 0.4067 | validation_loss = 0.3918 | val_acc=0.8045 | val_f1=0.6570 | val_top3=0.9948","\n","Epoch 3: train_loss = 0.3996 | validation_loss = 0.3941 | val_acc=0.7940 | val_f1=0.6794 | val_top3=0.9947","\n","Epoch 4: train_loss = 0.3969 | validation_loss = 0.3882 | val_acc=0.8070 | val_f1=0.6807 | val_top3=0.9950","\n","Epoch 5: train_loss = 0.3940 | validation_loss = 0.3876 | val_acc=0.8051 | val_f1=0.6638 | val_top3=0.9948","\n","Epoch 6: train_loss = 0.3921 | validation_loss = 0.3862 | val_acc=0.8046 | val_f1=0.6586 | val_top3=0.9950","\n","Epoch 7: train_loss = 0.3911 | validation_loss = 0.3861 | val_acc=0.8054 | val_f1=0.6663 | val_top3=0.9950","\n","Epoch 8: train_loss = 0.3901 | validation_loss = 0.3861 | val_acc=0.7966 | val_f1=0.6341 | val_top3=0.9945","\n","[test] loss=0.3886 acc=0.7938 macro_f1=0.6294 top3_acc=0.9940","\n","Execution time: 3 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"Below is a concise plan: The script loads experiment_data.npy from the working directory, parses the nested structure, and computes best or final metrics per dataset. It selects the best training loss (minimum over epochs), the best validation metrics (max accuracy/F1/top-3, min loss), and the final test metrics (only one entry). It then prints the dataset name followed by clearly labeled metrics and their values. The code executes immediately without any main-guard and does not create plots.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef load_experiment_data():\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    path = os.path.join(working_dir, \"experiment_data.npy\")\n    experiment_data = np.load(path, allow_pickle=True).item()\n    return experiment_data\n\n\ndef safe_min(records, key):\n    best_val = None\n    for r in records:\n        if key in r:\n            v = r[key]\n            if best_val is None or v < best_val:\n                best_val = v\n    return best_val\n\n\ndef safe_max(records, key):\n    best_val = None\n    for r in records:\n        if key in r:\n            v = r[key]\n            if best_val is None or v > best_val:\n                best_val = v\n    return best_val\n\n\ndef print_metrics(experiment_data):\n    for ds_name, ds in experiment_data.items():\n        print(f\"Dataset: {ds_name}\")\n\n        metrics = ds.get(\"metrics\", {})\n        train_metrics = metrics.get(\"train\", [])\n        val_metrics = metrics.get(\"val\", [])\n        test_metrics = metrics.get(\"test\", [])\n\n        # Best training loss (min over epochs)\n        train_loss_best = None\n        # In train list, original code saves entries like {\"epoch\": e, \"metric\": \"loss\", \"value\": train_loss, \"t\": ts}\n        # We'll pick min over \"value\" where metric == \"loss\"\n        train_loss_candidates = [\n            r for r in train_metrics if r.get(\"metric\") == \"loss\" and \"value\" in r\n        ]\n        if train_loss_candidates:\n            train_loss_best = min(r[\"value\"] for r in train_loss_candidates)\n\n        # Best validation metrics\n        # val entries have keys: accuracy, macro_f1, top3_accuracy, validation_loss\n        val_acc_best = safe_max(val_metrics, \"accuracy\")\n        val_f1_best = safe_max(val_metrics, \"macro_f1\")\n        val_top3_best = safe_max(val_metrics, \"top3_accuracy\")\n        val_loss_best = safe_min(val_metrics, \"validation_loss\")\n\n        # Final test metrics (there is typically one test entry appended at the end)\n        test_entry = test_metrics[-1] if test_metrics else {}\n        test_loss = test_entry.get(\"loss\")\n        test_acc = test_entry.get(\"accuracy\")\n        test_f1 = test_entry.get(\"macro_f1\")\n        test_top3 = test_entry.get(\"top3_accuracy\")\n\n        # Print with clear labels, only if available\n        if train_loss_best is not None:\n            print(f\"training loss (best): {train_loss_best}\")\n        if val_acc_best is not None:\n            print(f\"validation accuracy (best): {val_acc_best}\")\n        if val_f1_best is not None:\n            print(f\"validation macro F1 score (best): {val_f1_best}\")\n        if val_top3_best is not None:\n            print(f\"validation top-3 accuracy (best): {val_top3_best}\")\n        if val_loss_best is not None:\n            print(f\"validation loss (best): {val_loss_best}\")\n\n        if test_acc is not None:\n            print(f\"test accuracy: {test_acc}\")\n        if test_f1 is not None:\n            print(f\"test macro F1 score: {test_f1}\")\n        if test_top3 is not None:\n            print(f\"test top-3 accuracy: {test_top3}\")\n        if test_loss is not None:\n            print(f\"test loss: {test_loss}\")\n\n\ndef main():\n    data = load_experiment_data()\n    print_metrics(data)\n\n\n# Execute immediately without __main__ guard\nmain()\n","parse_term_out":["Dataset: BPI_base","\n","training loss (best): 0.390059571234457","\n","validation accuracy (best): 0.8069520252800919","\n","validation macro F1 score (best): 0.6806980145207687","\n","validation top-3 accuracy (best): 0.9950445294380188","\n","validation loss (best): 0.38610316197806543","\n","test accuracy: 0.7938350088704909","\n","test macro F1 score: 0.6294356158565257","\n","test top-3 accuracy: 0.9940124154090881","\n","test loss: 0.3886074724624742","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":208.31418251991272,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"","exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089","metric":{"value":{"metric_names":[{"metric_name":"training loss","lower_is_better":true,"description":"Loss computed on the training split; lower is better.","data":[{"dataset_name":"BPI_base","final_value":0.390059571234457,"best_value":0.390059571234457}]},{"metric_name":"validation accuracy","lower_is_better":false,"description":"Accuracy on the validation split; higher is better.","data":[{"dataset_name":"BPI_base","final_value":0.8069520252800919,"best_value":0.8069520252800919}]},{"metric_name":"validation macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the validation split; higher is better.","data":[{"dataset_name":"BPI_base","final_value":0.6806980145207687,"best_value":0.6806980145207687}]},{"metric_name":"validation top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the validation split; higher is better.","data":[{"dataset_name":"BPI_base","final_value":0.9950445294380188,"best_value":0.9950445294380188}]},{"metric_name":"validation loss","lower_is_better":true,"description":"Loss computed on the validation split; lower is better.","data":[{"dataset_name":"BPI_base","final_value":0.38610316197806543,"best_value":0.38610316197806543}]},{"metric_name":"test accuracy","lower_is_better":false,"description":"Accuracy on the held-out test split; higher is better.","data":[{"dataset_name":"BPI_base","final_value":0.7938350088704909,"best_value":0.7938350088704909}]},{"metric_name":"test macro F1 score","lower_is_better":false,"description":"Macro-averaged F1 score on the held-out test split; higher is better.","data":[{"dataset_name":"BPI_base","final_value":0.6294356158565257,"best_value":0.6294356158565257}]},{"metric_name":"test top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the held-out test split; higher is better.","data":[{"dataset_name":"BPI_base","final_value":0.9940124154090881,"best_value":0.9940124154090881}]},{"metric_name":"test loss","lower_is_better":true,"description":"Loss computed on the held-out test split; lower is better.","data":[{"dataset_name":"BPI_base","final_value":0.3886074724624742,"best_value":0.3886074724624742}]}]},"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":false,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":["../../logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_val_metrics.png","../../logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_confusion_matrix.png","../../logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_train_val_loss.png","../../logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_pr_curves.png"],"plot_paths":["experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_val_metrics.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_confusion_matrix.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_train_val_loss.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_pr_curves.png"],"plot_analyses":[{"analysis":"The validation metrics plot shows three key performance metrics: Validation Accuracy, Macro-F1, and Top-3 Accuracy. The validation accuracy is consistently high, around 80%, indicating that the model is performing well in predicting the next activity correctly most of the time. However, the Macro-F1 score is lower, starting around 65% and slightly increasing, suggesting that the class distribution might be imbalanced, affecting the harmonic mean of precision and recall. The Top-3 Accuracy remains constant at 100%, indicating that the correct next activity is almost always within the top three predicted activities.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_val_metrics.png"},{"analysis":"The confusion matrix provides insights into the model's performance across different classes. It shows the number of correct and incorrect predictions made by the model. A significant concentration along the diagonal suggests that many activities are correctly predicted, but some confusion exists between certain activities, as indicated by off-diagonal values. This could be due to similar activities or overlapping features among them.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_confusion_matrix.png"},{"analysis":"The training and validation loss plot indicates that both losses are decreasing over epochs, which is a positive sign. The training loss starts higher and decreases sharply, suggesting that the model is learning well from the training data. The validation loss also decreases but at a slower rate, reflecting a potential overfitting scenario where the model might be learning the training data better than the validation data. However, the close convergence of both losses towards the end suggests that the model is stabilizing and generalizing well.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_train_val_loss.png"},{"analysis":"The precision-recall curves plot shows high precision across different recall levels for micro-average, with an average precision (AP) of 0.927, which is excellent. The macro-average precision is lower (AP=0.724), indicating variability in model performance across different classes. This suggests that while the model is good at predicting the positive class, it might struggle with less frequent classes, highlighting the need for class balancing or additional feature engineering to improve predictions for minority classes.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_da7f619dc6be4f5fa90081d392370f2d_proc_332089/BPI2012_next_activity_pr_curves.png"}],"vlm_feedback_summary":"The experimental plots demonstrate a well-performing model with high accuracy and precision, especially in predicting the next activity in business processes. However, the analysis highlights potential class imbalance issues affecting the Macro-F1 score and macro-average precision, suggesting areas for improvement. The model shows a good balance between training and validation loss, and the high Top-3 Accuracy indicates reliable predictions within the top three options, though further exploration into class distribution might enhance overall performance.","datasets_successfully_tested":["['SYNTHETIC']"],"ablation_name":null,"hyperparam_name":null,"is_seed_node":false,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"# Set random seed\nimport random\nimport numpy as np\nimport torch\n\nseed = 0\nrandom.seed(seed)\nnp.random.seed(seed)\ntorch.manual_seed(seed)\nif torch.cuda.is_available():\n    torch.cuda.manual_seed(seed)\n\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nimport random\nimport math\nimport matplotlib.pyplot as plt\n\n# Device handling (required)\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data structure\nexperiment_data = {\n    \"BPI2012\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"BPI2017\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"ROAD\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n}\n\n# Data loading utilities (use provided helper)\nfrom ai_scientist.ideas.my_research_topic import load_datasets, pick_default_dataset\n\n\n# Reproducibility\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# Build prefixes\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    # Keep only 'complete' transitions if lifecycle exists\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        df = df[df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")]\n        if len(df) == 0:\n            df = df.copy()  # fallback if empty\n            df = df.sort_values([\"case_id\", \"timestamp\"])\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    # Build activity vocab\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}  # 0 for PAD\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        # Convert to numpy arrays for safe positional indexing\n        ts_ns = (\n            pd.to_datetime(g[\"timestamp\"], utc=True).astype(\"int64\").to_numpy()\n        )  # nanoseconds\n        ts = (ts_ns // 10**9).astype(np.int64)  # seconds as numpy array\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str).tolist()], dtype=np.int64\n        )\n        # simple calendar features\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        # time deltas and since start in seconds\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(\n            np.float32\n        )  # [T,5]\n        T = len(acts_ids)\n        if T < 2:\n            continue\n        # Generate prefixes of length k (min_prefix_len..min(max_prefix_len, T-1)); target = activity at position k\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            seq_acts = acts_ids[:k].tolist()\n            seq_feats = feats[:k]\n            target = int(acts_ids[k])\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": seq_acts,\n                    \"seq_feats\": seq_feats.copy(),\n                    \"target\": target,\n                    \"last_ts\": int(ts[k - 1]),\n                    \"next_ts\": int(ts[k]),\n                }\n            )\n\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n\n    # Collect feature normalization stats over all feats (initial; will be recomputed on train split)\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n    ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n    for s in samples:\n        if s[\"seq_feats\"].shape[0] > 0:\n            s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n            s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n    return samples, act2id, id2act, pad_id\n\n\n# Time-based split by case start time\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.zeros((pad_len, self.num_cont), dtype=np.float32)\n        feats_pad = np.vstack([feats_pad, feats.astype(np.float32)])\n        attn_mask = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad, dtype=torch.long),\n            \"feats\": torch.tensor(feats_pad, dtype=torch.float32),\n            \"mask\": torch.tensor(attn_mask, dtype=torch.float32),\n            \"y\": torch.tensor(s[\"target\"], dtype=torch.long),\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(\n        self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, num_layers=1, pad_idx=0\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim,\n            hidden_size=hidden,\n            batch_first=True,\n            num_layers=num_layers,\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)  # includes PAD index\n        self.pad_idx = pad_idx\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)  # [B,T,emb]\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h_last = h[-1]\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)\n        return logits\n\n\ndef collate_fn(batch):\n    keys = batch[0].keys()\n    out = {k: torch.stack([b[k] for b in batch], dim=0) for k in keys}\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys, preds_top1, preds_probs = [], [], []\n    top3_correct = 0\n    n_total = 0\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk_idx = torch.topk(probs, k=k_val, dim=1)\n            ys.extend(batch[\"y\"].detach().cpu().tolist())\n            preds_top1.extend(top1.detach().cpu().tolist())\n            preds_probs.append(probs.detach().cpu().numpy())\n            # top-3 correctness\n            for i in range(batch[\"y\"].size(0)):\n                if batch[\"y\"][i].item() in topk_idx[i].detach().cpu().tolist():\n                    top3_correct += 1\n            n_total += batch[\"y\"].size(0)\n    avg_loss = total_loss / max(1, n_total)\n    y_true = np.array(ys)\n    y_pred = np.array(preds_top1)\n    mask = y_true != pad_idx\n    y_true = y_true[mask]\n    y_pred = y_pred[mask]\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except Exception:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n_total))\n    probs_concat = (\n        np.concatenate(preds_probs, axis=0)\n        if len(preds_probs) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return avg_loss, acc, f1, top3, y_true, y_pred, probs_concat\n\n\ndef train_one_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    # Time-based split\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    # Build samples across all to get vocab; we'll re-normalize with train stats\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    # Filter per split\n    samples_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    samples_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    samples_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # Recompute normalization using train samples only\n    if len(samples_train) > 0:\n        concat_feats = [\n            s[\"seq_feats\"] for s in samples_train if s[\"seq_feats\"].shape[0] > 0\n        ]\n        if len(concat_feats) > 0:\n            all_feats = np.concatenate(concat_feats, axis=0)\n            dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n            ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n\n            def norm_samples(samples):\n                for s in samples:\n                    if s[\"seq_feats\"].shape[0] > 0:\n                        s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                        s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n            norm_samples(samples_train)\n            norm_samples(samples_val)\n            norm_samples(samples_test)\n    print(\n        f\"Samples train/val/test: {len(samples_train)}/{len(samples_val)}/{len(samples_test)}; vocab={len(act2id)}\"\n    )\n    if len(samples_train) == 0 or len(act2id) < 2:\n        print(\"Not enough data to train. Skipping.\")\n        return\n    ds_train = PrefixDataset(\n        samples_train, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_val = PrefixDataset(\n        samples_val, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_test = PrefixDataset(\n        samples_test, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    dl_train = DataLoader(\n        ds_train,\n        batch_size=batch_size,\n        shuffle=True,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_val = DataLoader(\n        ds_val,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_test = DataLoader(\n        ds_test,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n\n    # Model\n    model = LSTMBaseline(\n        vocab_size=len(act2id),\n        emb_dim=64,\n        cont_dim=5,\n        hidden=128,\n        num_layers=1,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss().to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n\n    # Training loop\n    best_val_top3 = -1.0\n    best_state = None\n    hist = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for epoch in range(1, max_epochs + 1):\n        model.train()\n        total = 0\n        running_loss = 0.0\n        for batch in dl_train:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            running_loss += loss.item() * logits.size(0)\n            total += logits.size(0)\n        train_loss = running_loss / max(1, total)\n        val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n            model, dl_val, criterion, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        hist[\"train_loss\"].append(train_loss)\n        hist[\"val_loss\"].append(val_loss)\n        hist[\"val_top3\"].append(val_top3)\n        experiment_data[name][\"losses\"][\"train\"].append((epoch, train_loss))\n        experiment_data[name][\"losses\"][\"val\"].append((epoch, val_loss))\n        experiment_data[name][\"metrics\"][\"val\"].append(\n            (epoch, {\"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3})\n        )\n        experiment_data[name][\"epochs\"].append(epoch)\n        if val_top3 > best_val_top3:\n            best_val_top3 = val_top3\n            best_state = {k: v.cpu().clone() for k, v in model.state_dict().items()}\n\n    # Load best\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n\n    # Final eval on train/val/test\n    train_loss, train_acc, train_f1, train_top3, _, _, _ = evaluate(\n        model, dl_train, criterion, device, len(act2id), pad_id\n    )\n    val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n        model, dl_val, criterion, device, len(act2id), pad_id\n    )\n    test_loss, test_acc, test_f1, test_top3, y_true_t, y_pred_t, probs_t = evaluate(\n        model, dl_test, criterion, device, len(act2id), pad_id\n    )\n    print(\n        f\"[{name}] Train: loss={train_loss:.4f} acc={train_acc:.4f} f1={train_f1:.4f} top3={train_top3:.4f}\"\n    )\n    print(\n        f\"[{name}] Test:  loss={test_loss:.4f} acc={test_acc:.4f} f1={test_f1:.4f} top3={test_top3:.4f}\"\n    )\n\n    # Save metrics\n    experiment_data[name][\"metrics\"][\"train\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": train_loss,\n                \"acc\": train_acc,\n                \"macro_f1\": train_f1,\n                \"top3\": train_top3,\n            },\n        )\n    )\n    experiment_data[name][\"metrics\"][\"val\"].append(\n        (\n            \"final\",\n            {\"loss\": val_loss, \"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3},\n        )\n    )\n    experiment_data[name][\"metrics\"][\"test\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": test_loss,\n                \"acc\": test_acc,\n                \"macro_f1\": test_f1,\n                \"top3\": test_top3,\n            },\n        )\n    )\n    experiment_data[name][\"predictions\"] = y_pred_t.tolist()\n    experiment_data[name][\"ground_truth\"] = y_true_t.tolist()\n\n    # Plots\n    try:\n        plt.figure()\n        plt.plot(hist[\"train_loss\"], label=\"train_loss\")\n        plt.plot(hist[\"val_loss\"], label=\"val_loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"loss_curves_{name}.png\"))\n        plt.close()\n\n        plt.figure()\n        plt.plot(hist[\"val_top3\"], label=\"val_top3\")\n        plt.legend()\n        plt.title(f\"Val Top-3 Acc - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Acc\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"val_top3_{name}.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n\n    # Save confusion matrix-like data (optional)\n    try:\n        from sklearn.metrics import confusion_matrix\n\n        cm = confusion_matrix(y_true_t, y_pred_t)\n        np.save(os.path.join(working_dir, f\"cm_{name}.npy\"), cm)\n    except Exception as e:\n        print(f\"[warn] Confusion matrix failed: {e}\")\n\n\ndef main():\n    datasets = load_datasets()\n    # Loop through loaded datasets; cap to 5000 earliest cases for speed\n    for key, df in datasets.items():\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep_cases = set(starts.iloc[:5000][\"case_id\"])\n                df_small = df[df[\"case_id\"].isin(keep_cases)].copy()\n            else:\n                df_small = df\n        except Exception:\n            df_small = df\n        train_one_dataset(\n            key, df_small, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    np.savez_compressed(\n        os.path.join(working_dir, \"experiment_data_compressed.npz\"),\n        data=experiment_data,\n    )\n\n\n# Execute immediately\nmain()\n","plan":"Seed node","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import (\n    f1_score,\n    accuracy_score,\n    confusion_matrix,\n    precision_recall_curve,\n    average_precision_score,\n)\nfrom sklearn.preprocessing import label_binarize\nimport random\nimport math\nfrom collections import defaultdict\n\n# Reproducibility and device\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# --------- Data loading from local XES via pm4py ----------\ndef load_xes_folder(data_dir=\"data\"):\n    datasets = {}\n    try:\n        import pm4py\n    except Exception as e:\n        print(f\"pm4py not available: {e}\")\n        return datasets\n    if not os.path.isdir(data_dir):\n        print(f\"Data directory not found: {data_dir}\")\n        return datasets\n    for fn in os.listdir(data_dir):\n        if fn.lower().endswith(\".xes\") or fn.lower().endswith(\".xes.gz\"):\n            path = os.path.join(data_dir, fn)\n            try:\n                log = pm4py.read_xes(path)\n                df = pm4py.convert_to_dataframe(log)\n                # Standardize columns\n                # pm4py dataframe typically has case:concept:name, concept:name, time:timestamp, lifecycle:transition\n                cols = df.columns\n                case_col = (\n                    \"case:concept:name\"\n                    if \"case:concept:name\" in cols\n                    else (\"case\" if \"case\" in cols else None)\n                )\n                act_col = (\n                    \"concept:name\"\n                    if \"concept:name\" in cols\n                    else (\"activity\" if \"activity\" in cols else None)\n                )\n                ts_col = (\n                    \"time:timestamp\"\n                    if \"time:timestamp\" in cols\n                    else (\"timestamp\" if \"timestamp\" in cols else None)\n                )\n                life_col = (\n                    \"lifecycle:transition\"\n                    if \"lifecycle:transition\" in cols\n                    else (\"lifecycle\" if \"lifecycle\" in cols else None)\n                )\n                if case_col is None or act_col is None or ts_col is None:\n                    print(f\"Missing required columns in {fn}, skipping.\")\n                    continue\n                out = pd.DataFrame(\n                    {\n                        \"case_id\": df[case_col].astype(str).values,\n                        \"activity\": df[act_col].astype(str).values,\n                        \"timestamp\": pd.to_datetime(df[ts_col], utc=True),\n                    }\n                )\n                if life_col is not None:\n                    out[\"lifecycle\"] = df[life_col].astype(str).values\n                name = os.path.splitext(fn)[0]\n                datasets[name] = out\n                print(\n                    f\"Loaded {name}: {len(out)} events, {out['case_id'].nunique()} cases\"\n                )\n            except Exception as e:\n                print(f\"Failed to load {fn}: {e}\")\n    return datasets\n\n\n# --------- Prefix building and split ----------\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        mask = df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")\n        if mask.any():\n            df = df[mask]\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        if len(g) < 2:\n            continue\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        ts = (g_ts.astype(\"int64\") // 10**9).to_numpy(np.int64)\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str)], dtype=np.int64\n        )\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(np.float32)\n        T = len(acts_ids)\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": acts_ids[:k].tolist(),\n                    \"seq_feats\": feats[:k].copy(),\n                    \"target\": int(acts_ids[k]),\n                    \"prefix_len\": k,\n                }\n            )\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    for s in samples:\n        pass  # initial no norm; will norm on train split\n    return samples, act2id, id2act, pad_id\n\n\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.vstack(\n            [\n                np.zeros((pad_len, self.num_cont), dtype=np.float32),\n                feats.astype(np.float32),\n            ]\n        )\n        attn = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad).long(),\n            \"feats\": torch.tensor(feats_pad).float(),\n            \"mask\": torch.tensor(attn).float(),\n            \"y\": torch.tensor(s[\"target\"]).long(),\n            \"prefix_len\": L,\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, pad_idx=0):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim, hidden_size=hidden, batch_first=True\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h = self.dropout(h[-1])\n        return self.fc(h)\n\n\ndef collate_fn(batch):\n    out = {\n        k: (\n            torch.stack([b[k] for b in batch], 0)\n            if isinstance(batch[0][k], torch.Tensor)\n            else [b[k] for b in batch]\n        )\n        for k in batch[0].keys()\n    }\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys = []\n    yhat = []\n    probs_list = []\n    n = 0\n    top3_correct = 0\n    pref_lens = []\n    top3_flags = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk = torch.topk(probs, k=k_val, dim=1)\n            y = batch[\"y\"]\n            ys.extend(y.detach().cpu().tolist())\n            yhat.extend(top1.detach().cpu().tolist())\n            probs_list.append(probs.detach().cpu().numpy())\n            for i in range(y.size(0)):\n                flag = int(y[i].item() in topk[i].detach().cpu().tolist())\n                top3_correct += flag\n                top3_flags.append(flag)\n                pref_lens.append(int(batch[\"prefix_len\"][i].item()))\n            n += y.size(0)\n    avg_loss = total_loss / max(1, n)\n    y_true = np.array(ys)\n    y_pred = np.array(yhat)\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n))\n    probs_concat = (\n        np.concatenate(probs_list, axis=0)\n        if len(probs_list) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return (\n        avg_loss,\n        acc,\n        f1,\n        top3,\n        y_true,\n        y_pred,\n        probs_concat,\n        np.array(pref_lens),\n        np.array(top3_flags),\n    )\n\n\ndef train_on_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    s_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    s_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    s_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # normalize time features on train\n    if len(s_train) > 0:\n        feats = np.concatenate(\n            [s[\"seq_feats\"] for s in s_train if len(s[\"seq_feats\"]) > 0], axis=0\n        )\n        dt_mean, dt_std = feats[:, 0].mean(), feats[:, 0].std() + 1e-6\n        ss_mean, ss_std = feats[:, 1].mean(), feats[:, 1].std() + 1e-6\n\n        def norm(samples):\n            for s in samples:\n                if s[\"seq_feats\"].shape[0] > 0:\n                    s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                    s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n        norm(s_train)\n        norm(s_val)\n        norm(s_test)\n    print(\n        f\"Samples train/val/test: {len(s_train)}/{len(s_val)}/{len(s_test)}; vocab={len(act2id)}\"\n    )\n    if len(s_train) == 0 or len(act2id) < 2:\n        print(\"Insufficient data; skipping.\")\n        return None\n    ds_tr = PrefixDataset(s_train, pad_id, max_prefix_len, 5)\n    ds_va = PrefixDataset(s_val, pad_id, max_prefix_len, 5)\n    ds_te = PrefixDataset(s_test, pad_id, max_prefix_len, 5)\n    dl_tr = DataLoader(\n        ds_tr, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n    )\n    dl_va = DataLoader(\n        ds_va, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    dl_te = DataLoader(\n        ds_te, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    model = LSTMBaseline(\n        vocab_size=len(act2id), emb_dim=64, cont_dim=5, hidden=128, pad_idx=pad_id\n    ).to(device)\n    crit = nn.CrossEntropyLoss().to(device)\n    opt = torch.optim.Adam(model.parameters(), lr=lr)\n    best_top3 = -1.0\n    best_state = None\n    history = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for ep in range(1, max_epochs + 1):\n        model.train()\n        tot = 0\n        run_loss = 0.0\n        for batch in dl_tr:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            opt.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = crit(logits, batch[\"y\"])\n            loss.backward()\n            opt.step()\n            run_loss += loss.item() * logits.size(0)\n            tot += logits.size(0)\n        tr_loss = run_loss / max(1, tot)\n        va_loss, va_acc, va_f1, va_top3, *_ = evaluate(\n            model, dl_va, crit, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {ep}: val_loss={va_loss:.4f} acc={va_acc:.4f} f1={va_f1:.4f} top3={va_top3:.4f}\"\n        )\n        history[\"train_loss\"].append(tr_loss)\n        history[\"val_loss\"].append(va_loss)\n        history[\"val_top3\"].append(va_top3)\n        if va_top3 > best_top3:\n            best_top3 = va_top3\n            best_state = {\n                k: v.detach().cpu().clone() for k, v in model.state_dict().items()\n            }\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n    tr_loss, tr_acc, tr_f1, tr_top3, *_ = evaluate(\n        model, dl_tr, crit, device, len(act2id), pad_id\n    )\n    te_loss, te_acc, te_f1, te_top3, y_true, y_pred, probs, pref_lens, top3_flags = (\n        evaluate(model, dl_te, crit, device, len(act2id), pad_id)\n    )\n    print(\n        f\"[{name}] Test: loss={te_loss:.4f} acc={te_acc:.4f} f1={te_f1:.4f} top3={te_top3:.4f}\"\n    )\n    # package experiment data\n    exp = {\n        \"metrics\": {\n            \"train\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": tr_loss,\n                        \"acc\": tr_acc,\n                        \"macro_f1\": tr_f1,\n                        \"top3\": tr_top3,\n                    },\n                )\n            ],\n            \"val\": [],\n            \"test\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": te_loss,\n                        \"acc\": te_acc,\n                        \"macro_f1\": te_f1,\n                        \"top3\": te_top3,\n                    },\n                )\n            ],\n        },\n        \"losses\": {\n            \"train\": list(enumerate(history[\"train_loss\"], start=1)),\n            \"val\": list(enumerate(history[\"val_loss\"], start=1)),\n        },\n        \"predictions\": y_pred.tolist(),\n        \"ground_truth\": y_true.tolist(),\n        \"epochs\": list(range(1, len(history[\"train_loss\"]) + 1)),\n        \"probs\": probs,\n        \"prefix_lens\": pref_lens.tolist(),\n        \"top3_flags\": top3_flags.tolist(),\n        \"act2id\": act2id,\n    }\n    # plots for this dataset\n    try:\n        plt.figure()\n        plt.plot(history[\"train_loss\"], label=\"train\")\n        plt.plot(history[\"val_loss\"], label=\"val\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating loss curves for {name}: {e}\")\n        plt.close()\n    try:\n        cm = confusion_matrix(y_true, y_pred)\n        plt.figure(figsize=(5, 4))\n        plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n        plt.colorbar()\n        plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n        plt.xlabel(\"Predicted\")\n        plt.ylabel(\"True\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_confusion_matrix.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating confusion matrix for {name}: {e}\")\n        plt.close()\n    try:\n        # Top-3 accuracy vs prefix length\n        if len(pref_lens) > 0:\n            d = defaultdict(list)\n            for L, flag in zip(pref_lens, top3_flags):\n                d[int(L)].append(int(flag))\n            xs = sorted(d.keys())\n            ys = [np.mean(d[k]) for k in xs]\n            plt.figure()\n            plt.plot(xs, ys, marker=\"o\")\n            plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n            plt.xlabel(\"Prefix Length\")\n            plt.ylabel(\"Top-3 Accuracy\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_top3_vs_prefixlen.png\"))\n            plt.close()\n    except Exception as e:\n        print(f\"Error creating Top-3 vs prefix length for {name}: {e}\")\n        plt.close()\n    try:\n        # Macro PR curve (one-vs-rest); may be coarse due to many classes\n        if probs.shape[0] > 0:\n            classes = np.unique(y_true)\n            Y = label_binarize(y_true, classes=range(probs.shape[1]))\n            # only keep columns present in classes to avoid PAD\n            present = [c for c in classes]\n            if len(present) > 1:\n                precisions = []\n                recalls = []\n                aps = []\n                for c in present:\n                    p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                    ap = average_precision_score(Y[:, c], probs[:, c])\n                    precisions.append(\n                        np.interp(np.linspace(0, 1, 101), r[::-1], p[::-1])\n                    )\n                    recalls.append(np.linspace(0, 1, 101))\n                    aps.append(ap)\n                macro_p = np.mean(np.stack(precisions, 0), 0)\n                macro_r = np.linspace(0, 1, 101)\n                plt.figure()\n                plt.plot(macro_r, macro_p, label=f\"Macro-PR (mAP={np.mean(aps):.3f})\")\n                plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Recall\")\n                plt.ylabel(\"Precision\")\n                plt.legend()\n                plt.tight_layout()\n                plt.savefig(os.path.join(working_dir, f\"{name}_macro_pr.png\"))\n                plt.close()\n    except Exception as e:\n        print(f\"Error creating PR curve for {name}: {e}\")\n        plt.close()\n    return name, exp\n\n\ndef main():\n    datasets = load_xes_folder(data_dir=os.path.join(os.getcwd(), \"data\"))\n    experiment_data = {}\n    for name, df in datasets.items():\n        # optional cap earliest 5000 cases\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep = set(starts.iloc[:5000][\"case_id\"])\n                df = df[df[\"case_id\"].isin(keep)].copy()\n        except:\n            pass\n        res = train_on_dataset(\n            name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n        if res is not None:\n            k, exp = res\n            experiment_data[k] = exp\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Print evaluation metrics\n    for k, v in experiment_data.items():\n        test_metrics = dict(v[\"metrics\"][\"test\"][0][1])\n        print(\n            f\"{k} | Test acc={test_metrics['acc']:.4f} macro_f1={test_metrics['macro_f1']:.4f} top3={test_metrics['top3']:.4f} loss={test_metrics['loss']:.4f}\"\n        )\n\n    # Secondary plotting pass strictly from experiment_data.npy (as required)\n    try:\n        experiment_data_loaded = np.load(\n            os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n        ).item()\n    except Exception as e:\n        print(f\"Error loading experiment data: {e}\")\n        experiment_data_loaded = {}\n    for name, ed in experiment_data_loaded.items():\n        try:\n            # re-plot loss curves from saved data\n            plt.figure()\n            tl = [y for (_, y) in ed.get(\"losses\", {}).get(\"train\", [])]\n            vl = [y for (_, y) in ed.get(\"losses\", {}).get(\"val\", [])]\n            if len(tl) > 0:\n                plt.plot(tl, label=\"train\")\n            if len(vl) > 0:\n                plt.plot(vl, label=\"val\")\n            plt.legend()\n            plt.title(f\"Loss Curves - {name}\\nNext-activity\")\n            plt.xlabel(\"Epoch\")\n            plt.ylabel(\"Loss\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves_reload.png\"))\n            plt.close()\n        except Exception as e:\n            print(f\"Error creating plot1: {e}\")\n            plt.close()\n        try:\n            # confusion matrix from predictions and ground truth\n            y_true = ed.get(\"ground_truth\", [])\n            y_pred = ed.get(\"predictions\", [])\n            if len(y_true) > 0 and len(y_pred) > 0:\n                cm = confusion_matrix(y_true, y_pred)\n                plt.figure(figsize=(5, 4))\n                plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n                plt.colorbar()\n                plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Predicted\")\n                plt.ylabel(\"True\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_confusion_matrix_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot2: {e}\")\n            plt.close()\n        try:\n            # Top-3 vs prefix length if present\n            pref = ed.get(\"prefix_lens\", [])\n            flags = ed.get(\"top3_flags\", [])\n            if len(pref) > 0 and len(flags) > 0:\n                d = defaultdict(list)\n                for L, f in zip(pref, flags):\n                    d[int(L)].append(int(f))\n                xs = sorted(d.keys())\n                ys = [float(np.mean(d[x])) for x in xs]\n                plt.figure()\n                plt.plot(xs, ys, marker=\"o\")\n                plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n                plt.xlabel(\"Prefix Length\")\n                plt.ylabel(\"Top-3 Accuracy\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_top3_vs_prefixlen_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot3: {e}\")\n            plt.close()\n        try:\n            # Macro PR curve if probs available\n            probs = np.array(ed.get(\"probs\", []))\n            y_true = ed.get(\"ground_truth\", [])\n            if probs.size > 0 and len(y_true) > 0:\n                classes = sorted(set(y_true))\n                Y = label_binarize(np.array(y_true), classes=range(probs.shape[1]))\n                present = classes\n                if len(present) > 1:\n                    precisions = []\n                    aps = []\n                    grid = np.linspace(0, 1, 101)\n                    for c in present:\n                        p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                        precisions.append(np.interp(grid, r[::-1], p[::-1]))\n                        aps.append(average_precision_score(Y[:, c], probs[:, c]))\n                    macro_p = np.mean(np.stack(precisions, 0), 0)\n                    plt.figure()\n                    plt.plot(grid, macro_p, label=f\"mAP={np.mean(aps):.3f}\")\n                    plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                    plt.xlabel(\"Recall\")\n                    plt.ylabel(\"Precision\")\n                    plt.legend()\n                    plt.tight_layout()\n                    plt.savefig(\n                        os.path.join(working_dir, f\"{name}_macro_pr_reload.png\")\n                    )\n                    plt.close()\n        except Exception as e:\n            print(f\"Error creating plot4: {e}\")\n            plt.close()\n\n\nif __name__ == \"__main__\":\n    main()\n","plot_plan":null,"step":6,"id":"9a2ff2eda58e4ea8966e9f8f6f6ba8c5","ctime":1757758298.9803598,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available in /workspace/data: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<22:28,  9.71it/s]","\rparsing log, completed traces ::   1%|1         | 153/13087 [00:00<00:14, 886.38it/s]","\rparsing log, completed traces ::   2%|2         | 303/13087 [00:00<00:10, 1164.46it/s]","\rparsing log, completed traces ::   3%|3         | 456/13087 [00:00<00:09, 1306.74it/s]","\rparsing log, completed traces ::   5%|4         | 618/13087 [00:00<00:08, 1418.78it/s]","\rparsing log, completed traces ::   6%|6         | 797/13087 [00:00<00:07, 1541.75it/s]","\rparsing log, completed traces ::   7%|7         | 957/13087 [00:00<00:07, 1556.20it/s]","\rparsing log, completed traces ::   9%|8         | 1115/13087 [00:00<00:07, 1556.86it/s]","\rparsing log, completed traces ::  10%|9         | 1277/13087 [00:00<00:07, 1575.37it/s]","\rparsing log, completed traces ::  11%|#         | 1435/13087 [00:01<00:14, 826.79it/s] ","\rparsing log, completed traces ::  12%|#2        | 1612/13087 [00:01<00:11, 1002.34it/s]","\rparsing log, completed traces ::  14%|#3        | 1782/13087 [00:01<00:09, 1148.64it/s]","\rparsing log, completed traces ::  15%|#4        | 1930/13087 [00:01<00:09, 1219.98it/s]","\rparsing log, completed traces ::  16%|#5        | 2093/13087 [00:01<00:08, 1320.70it/s]","\rparsing log, completed traces ::  17%|#7        | 2251/13087 [00:01<00:07, 1383.36it/s]","\rparsing log, completed traces ::  19%|#8        | 2426/13087 [00:01<00:07, 1480.42it/s]","\rparsing log, completed traces ::  20%|#9        | 2602/13087 [00:02<00:06, 1555.00it/s]","\rparsing log, completed traces ::  21%|##1       | 2766/13087 [00:02<00:06, 1548.24it/s]","\rparsing log, completed traces ::  22%|##2       | 2927/13087 [00:02<00:06, 1541.96it/s]","\rparsing log, completed traces ::  24%|##3       | 3086/13087 [00:02<00:06, 1534.60it/s]","\rparsing log, completed traces ::  25%|##5       | 3288/13087 [00:02<00:05, 1673.71it/s]","\rparsing log, completed traces ::  27%|##6       | 3505/13087 [00:02<00:05, 1817.96it/s]","\rparsing log, completed traces ::  28%|##8       | 3690/13087 [00:02<00:05, 1806.38it/s]","\rparsing log, completed traces ::  30%|##9       | 3873/13087 [00:02<00:05, 1753.89it/s]","\rparsing log, completed traces ::  31%|###1      | 4085/13087 [00:02<00:04, 1859.02it/s]","\rparsing log, completed traces ::  33%|###2      | 4273/13087 [00:03<00:08, 1000.62it/s]","\rparsing log, completed traces ::  34%|###4      | 4460/13087 [00:03<00:07, 1159.18it/s]","\rparsing log, completed traces ::  36%|###5      | 4674/13087 [00:03<00:06, 1361.00it/s]","\rparsing log, completed traces ::  37%|###7      | 4850/13087 [00:03<00:05, 1450.68it/s]","\rparsing log, completed traces ::  39%|###8      | 5041/13087 [00:03<00:05, 1557.36it/s]","\rparsing log, completed traces ::  40%|###9      | 5220/13087 [00:03<00:04, 1580.66it/s]","\rparsing log, completed traces ::  41%|####1     | 5396/13087 [00:03<00:04, 1628.13it/s]","\rparsing log, completed traces ::  43%|####2     | 5571/13087 [00:03<00:04, 1566.84it/s]","\rparsing log, completed traces ::  44%|####3     | 5737/13087 [00:04<00:04, 1576.83it/s]","\rparsing log, completed traces ::  45%|####5     | 5915/13087 [00:04<00:04, 1629.25it/s]","\rparsing log, completed traces ::  46%|####6     | 6083/13087 [00:04<00:04, 1624.25it/s]","\rparsing log, completed traces ::  48%|####7     | 6257/13087 [00:04<00:04, 1656.50it/s]","\rparsing log, completed traces ::  49%|####9     | 6452/13087 [00:04<00:03, 1740.42it/s]","\rparsing log, completed traces ::  51%|#####     | 6629/13087 [00:04<00:03, 1671.00it/s]","\rparsing log, completed traces ::  52%|#####1    | 6799/13087 [00:04<00:03, 1647.18it/s]","\rparsing log, completed traces ::  53%|#####3    | 6966/13087 [00:04<00:03, 1571.29it/s]","\rparsing log, completed traces ::  55%|#####4    | 7138/13087 [00:04<00:03, 1601.56it/s]","\rparsing log, completed traces ::  56%|#####5    | 7303/13087 [00:05<00:03, 1610.29it/s]","\rparsing log, completed traces ::  57%|#####7    | 7465/13087 [00:05<00:07, 778.90it/s] ","\rparsing log, completed traces ::  58%|#####8    | 7620/13087 [00:05<00:06, 904.28it/s]","\rparsing log, completed traces ::  59%|#####9    | 7779/13087 [00:05<00:05, 1034.57it/s]","\rparsing log, completed traces ::  61%|######    | 7925/13087 [00:05<00:04, 1125.05it/s]","\rparsing log, completed traces ::  62%|######1   | 8091/13087 [00:05<00:03, 1249.13it/s]","\rparsing log, completed traces ::  63%|######3   | 8263/13087 [00:05<00:03, 1366.33it/s]","\rparsing log, completed traces ::  64%|######4   | 8425/13087 [00:06<00:03, 1428.82it/s]","\rparsing log, completed traces ::  66%|######5   | 8607/13087 [00:06<00:02, 1535.10it/s]","\rparsing log, completed traces ::  67%|######7   | 8773/13087 [00:06<00:02, 1569.29it/s]","\rparsing log, completed traces ::  68%|######8   | 8938/13087 [00:06<00:02, 1572.65it/s]","\rparsing log, completed traces ::  70%|######9   | 9101/13087 [00:06<00:02, 1543.41it/s]","\rparsing log, completed traces ::  71%|#######   | 9285/13087 [00:06<00:02, 1619.38it/s]","\rparsing log, completed traces ::  72%|#######2  | 9454/13087 [00:06<00:02, 1639.11it/s]","\rparsing log, completed traces ::  74%|#######3  | 9621/13087 [00:06<00:02, 1577.07it/s]","\rparsing log, completed traces ::  75%|#######4  | 9787/13087 [00:06<00:02, 1599.63it/s]","\rparsing log, completed traces ::  76%|#######6  | 9950/13087 [00:07<00:01, 1607.35it/s]","\rparsing log, completed traces ::  77%|#######7  | 10112/13087 [00:07<00:01, 1599.83it/s]","\rparsing log, completed traces ::  78%|#######8  | 10273/13087 [00:07<00:01, 1570.05it/s]","\rparsing log, completed traces ::  80%|#######9  | 10431/13087 [00:07<00:01, 1531.82it/s]","\rparsing log, completed traces ::  81%|########  | 10585/13087 [00:07<00:01, 1478.04it/s]","\rparsing log, completed traces ::  82%|########2 | 10761/13087 [00:07<00:01, 1556.07it/s]","\rparsing log, completed traces ::  83%|########3 | 10923/13087 [00:08<00:02, 738.99it/s] ","\rparsing log, completed traces ::  85%|########4 | 11088/13087 [00:08<00:02, 886.03it/s]","\rparsing log, completed traces ::  86%|########6 | 11264/13087 [00:08<00:01, 1048.75it/s]","\rparsing log, completed traces ::  88%|########7 | 11463/13087 [00:08<00:01, 1246.06it/s]","\rparsing log, completed traces ::  89%|########8 | 11630/13087 [00:08<00:01, 1343.80it/s]","\rparsing log, completed traces ::  90%|######### | 11831/13087 [00:08<00:00, 1508.01it/s]","\rparsing log, completed traces ::  92%|#########1| 12015/13087 [00:08<00:00, 1592.47it/s]","\rparsing log, completed traces ::  93%|#########3| 12203/13087 [00:08<00:00, 1669.88it/s]","\rparsing log, completed traces ::  95%|#########4| 12400/13087 [00:08<00:00, 1751.54it/s]","\rparsing log, completed traces ::  96%|#########6| 12601/13087 [00:08<00:00, 1823.25it/s]","\rparsing log, completed traces ::  98%|#########7| 12791/13087 [00:09<00:00, 1826.19it/s]","\rparsing log, completed traces ::  99%|#########9| 12981/13087 [00:09<00:00, 1844.94it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:09<00:00, 1420.94it/s]","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<4:39:26,  1.88it/s]","\rparsing log, completed traces ::   0%|          | 66/31509 [00:00<03:46, 138.64it/s]","\rparsing log, completed traces ::   0%|          | 121/31509 [00:00<02:14, 233.35it/s]","\rparsing log, completed traces ::   1%|          | 184/31509 [00:00<01:34, 330.57it/s]","\rparsing log, completed traces ::   1%|          | 244/31509 [00:00<01:18, 400.27it/s]","\rparsing log, completed traces ::   1%|          | 305/31509 [00:01<01:08, 456.89it/s]","\rparsing log, completed traces ::   1%|1         | 364/31509 [00:01<01:03, 492.80it/s]","\rparsing log, completed traces ::   1%|1         | 427/31509 [00:01<00:58, 530.03it/s]","\rparsing log, completed traces ::   2%|1         | 486/31509 [00:01<00:56, 547.04it/s]","\rparsing log, completed traces ::   2%|1         | 545/31509 [00:01<00:56, 552.62it/s]","\rparsing log, completed traces ::   2%|1         | 607/31509 [00:01<00:54, 571.44it/s]","\rparsing log, completed traces ::   2%|2         | 676/31509 [00:01<00:50, 605.53it/s]","\rparsing log, completed traces ::   2%|2         | 739/31509 [00:01<00:50, 608.77it/s]","\rparsing log, completed traces ::   3%|2         | 801/31509 [00:01<00:50, 605.76it/s]","\rparsing log, completed traces ::   3%|2         | 863/31509 [00:01<00:50, 601.22it/s]","\rparsing log, completed traces ::   3%|2         | 924/31509 [00:02<01:36, 317.07it/s]","\rparsing log, completed traces ::   3%|3         | 992/31509 [00:02<01:19, 381.75it/s]","\rparsing log, completed traces ::   3%|3         | 1051/31509 [00:02<01:12, 422.46it/s]","\rparsing log, completed traces ::   4%|3         | 1114/31509 [00:02<01:04, 469.13it/s]","\rparsing log, completed traces ::   4%|3         | 1180/31509 [00:02<00:58, 514.61it/s]","\rparsing log, completed traces ::   4%|3         | 1248/31509 [00:02<00:54, 557.07it/s]","\rparsing log, completed traces ::   4%|4         | 1311/31509 [00:02<00:53, 565.62it/s]","\rparsing log, completed traces ::   4%|4         | 1381/31509 [00:03<00:50, 602.31it/s]","\rparsing log, completed traces ::   5%|4         | 1445/31509 [00:03<00:51, 587.19it/s]","\rparsing log, completed traces ::   5%|4         | 1507/31509 [00:03<00:50, 591.62it/s]","\rparsing log, completed traces ::   5%|4         | 1571/31509 [00:03<00:49, 602.27it/s]","\rparsing log, completed traces ::   5%|5         | 1634/31509 [00:03<00:49, 609.48it/s]","\rparsing log, completed traces ::   5%|5         | 1696/31509 [00:03<00:49, 601.55it/s]","\rparsing log, completed traces ::   6%|5         | 1758/31509 [00:03<00:49, 606.22it/s]","\rparsing log, completed traces ::   6%|5         | 1820/31509 [00:03<00:50, 592.98it/s]","\rparsing log, completed traces ::   6%|5         | 1880/31509 [00:03<00:50, 585.23it/s]","\rparsing log, completed traces ::   6%|6         | 1944/31509 [00:04<00:49, 597.78it/s]","\rparsing log, completed traces ::   6%|6         | 2007/31509 [00:04<00:48, 605.79it/s]","\rparsing log, completed traces ::   7%|6         | 2068/31509 [00:04<01:33, 314.84it/s]","\rparsing log, completed traces ::   7%|6         | 2132/31509 [00:04<01:18, 372.38it/s]","\rparsing log, completed traces ::   7%|6         | 2195/31509 [00:04<01:09, 424.09it/s]","\rparsing log, completed traces ::   7%|7         | 2262/31509 [00:04<01:01, 478.77it/s]","\rparsing log, completed traces ::   7%|7         | 2326/31509 [00:04<00:56, 516.64it/s]","\rparsing log, completed traces ::   8%|7         | 2388/31509 [00:05<00:53, 540.26it/s]","\rparsing log, completed traces ::   8%|7         | 2453/31509 [00:05<00:51, 566.55it/s]","\rparsing log, completed traces ::   8%|7         | 2516/31509 [00:05<00:49, 583.83it/s]","\rparsing log, completed traces ::   8%|8         | 2587/31509 [00:05<00:46, 615.57it/s]","\rparsing log, completed traces ::   8%|8         | 2655/31509 [00:05<00:45, 630.27it/s]","\rparsing log, completed traces ::   9%|8         | 2720/31509 [00:05<00:45, 626.34it/s]","\rparsing log, completed traces ::   9%|8         | 2789/31509 [00:05<00:44, 643.39it/s]","\rparsing log, completed traces ::   9%|9         | 2859/31509 [00:05<00:43, 658.38it/s]","\rparsing log, completed traces ::   9%|9         | 2928/31509 [00:05<00:42, 665.86it/s]","\rparsing log, completed traces ::  10%|9         | 2999/31509 [00:05<00:42, 677.93it/s]","\rparsing log, completed traces ::  10%|9         | 3068/31509 [00:06<00:43, 657.74it/s]","\rparsing log, completed traces ::  10%|9         | 3135/31509 [00:06<00:44, 644.62it/s]","\rparsing log, completed traces ::  10%|#         | 3200/31509 [00:06<00:44, 632.81it/s]","\rparsing log, completed traces ::  10%|#         | 3270/31509 [00:06<00:43, 650.45it/s]","\rparsing log, completed traces ::  11%|#         | 3342/31509 [00:06<00:42, 668.86it/s]","\rparsing log, completed traces ::  11%|#         | 3412/31509 [00:06<00:41, 674.83it/s]","\rparsing log, completed traces ::  11%|#1        | 3480/31509 [00:07<01:26, 323.27it/s]","\rparsing log, completed traces ::  11%|#1        | 3545/31509 [00:07<01:14, 376.93it/s]","\rparsing log, completed traces ::  11%|#1        | 3615/31509 [00:07<01:03, 438.70it/s]","\rparsing log, completed traces ::  12%|#1        | 3683/31509 [00:07<00:56, 488.47it/s]","\rparsing log, completed traces ::  12%|#1        | 3751/31509 [00:07<00:52, 532.69it/s]","\rparsing log, completed traces ::  12%|#2        | 3815/31509 [00:07<00:49, 556.59it/s]","\rparsing log, completed traces ::  12%|#2        | 3881/31509 [00:07<00:47, 582.92it/s]","\rparsing log, completed traces ::  13%|#2        | 3948/31509 [00:07<00:45, 605.04it/s]","\rparsing log, completed traces ::  13%|#2        | 4018/31509 [00:07<00:43, 629.22it/s]","\rparsing log, completed traces ::  13%|#2        | 4085/31509 [00:07<00:43, 635.34it/s]","\rparsing log, completed traces ::  13%|#3        | 4153/31509 [00:08<00:42, 647.23it/s]","\rparsing log, completed traces ::  13%|#3        | 4220/31509 [00:08<00:42, 639.47it/s]","\rparsing log, completed traces ::  14%|#3        | 4286/31509 [00:08<00:43, 628.02it/s]","\rparsing log, completed traces ::  14%|#3        | 4351/31509 [00:08<00:42, 632.63it/s]","\rparsing log, completed traces ::  14%|#4        | 4419/31509 [00:08<00:42, 644.38it/s]","\rparsing log, completed traces ::  14%|#4        | 4484/31509 [00:08<00:42, 632.32it/s]","\rparsing log, completed traces ::  14%|#4        | 4552/31509 [00:08<00:41, 642.19it/s]","\rparsing log, completed traces ::  15%|#4        | 4618/31509 [00:08<00:41, 645.88it/s]","\rparsing log, completed traces ::  15%|#4        | 4687/31509 [00:08<00:40, 656.46it/s]","\rparsing log, completed traces ::  15%|#5        | 4761/31509 [00:09<00:39, 678.85it/s]","\rparsing log, completed traces ::  15%|#5        | 4830/31509 [00:09<00:39, 669.27it/s]","\rparsing log, completed traces ::  16%|#5        | 4898/31509 [00:09<00:40, 662.59it/s]","\rparsing log, completed traces ::  16%|#5        | 4965/31509 [00:09<01:26, 306.46it/s]","\rparsing log, completed traces ::  16%|#5        | 5033/31509 [00:09<01:12, 366.63it/s]","\rparsing log, completed traces ::  16%|#6        | 5103/31509 [00:09<01:01, 427.56it/s]","\rparsing log, completed traces ::  16%|#6        | 5166/31509 [00:10<00:56, 468.61it/s]","\rparsing log, completed traces ::  17%|#6        | 5234/31509 [00:10<00:50, 516.91it/s]","\rparsing log, completed traces ::  17%|#6        | 5300/31509 [00:10<00:47, 551.86it/s]","\rparsing log, completed traces ::  17%|#7        | 5366/31509 [00:10<00:45, 579.01it/s]","\rparsing log, completed traces ::  17%|#7        | 5431/31509 [00:10<00:44, 591.70it/s]","\rparsing log, completed traces ::  17%|#7        | 5501/31509 [00:10<00:41, 620.87it/s]","\rparsing log, completed traces ::  18%|#7        | 5569/31509 [00:10<00:40, 637.35it/s]","\rparsing log, completed traces ::  18%|#7        | 5643/31509 [00:10<00:38, 665.91it/s]","\rparsing log, completed traces ::  18%|#8        | 5717/31509 [00:10<00:37, 683.80it/s]","\rparsing log, completed traces ::  18%|#8        | 5787/31509 [00:10<00:37, 681.81it/s]","\rparsing log, completed traces ::  19%|#8        | 5857/31509 [00:11<00:37, 685.34it/s]","\rparsing log, completed traces ::  19%|#8        | 5932/31509 [00:11<00:36, 699.50it/s]","\rparsing log, completed traces ::  19%|#9        | 6003/31509 [00:11<00:37, 686.90it/s]","\rparsing log, completed traces ::  19%|#9        | 6073/31509 [00:11<00:37, 672.78it/s]","\rparsing log, completed traces ::  19%|#9        | 6141/31509 [00:11<00:37, 672.00it/s]","\rparsing log, completed traces ::  20%|#9        | 6210/31509 [00:11<00:37, 674.37it/s]","\rparsing log, completed traces ::  20%|#9        | 6278/31509 [00:11<00:37, 672.18it/s]","\rparsing log, completed traces ::  20%|##        | 6346/31509 [00:11<00:38, 662.15it/s]","\rparsing log, completed traces ::  20%|##        | 6417/31509 [00:11<00:37, 673.43it/s]","\rparsing log, completed traces ::  21%|##        | 6485/31509 [00:11<00:37, 664.52it/s]","\rparsing log, completed traces ::  21%|##        | 6552/31509 [00:12<00:39, 635.34it/s]","\rparsing log, completed traces ::  21%|##        | 6616/31509 [00:12<00:39, 631.31it/s]","\rparsing log, completed traces ::  21%|##1       | 6683/31509 [00:12<00:38, 640.26it/s]","\rparsing log, completed traces ::  21%|##1       | 6748/31509 [00:12<01:32, 268.44it/s]","\rparsing log, completed traces ::  22%|##1       | 6814/31509 [00:12<01:15, 326.06it/s]","\rparsing log, completed traces ::  22%|##1       | 6878/31509 [00:13<01:04, 380.45it/s]","\rparsing log, completed traces ::  22%|##2       | 6946/31509 [00:13<00:55, 439.54it/s]","\rparsing log, completed traces ::  22%|##2       | 7015/31509 [00:13<00:49, 494.57it/s]","\rparsing log, completed traces ::  22%|##2       | 7085/31509 [00:13<00:44, 543.74it/s]","\rparsing log, completed traces ::  23%|##2       | 7153/31509 [00:13<00:42, 575.66it/s]","\rparsing log, completed traces ::  23%|##2       | 7219/31509 [00:13<00:41, 582.03it/s]","\rparsing log, completed traces ::  23%|##3       | 7285/31509 [00:13<00:40, 602.81it/s]","\rparsing log, completed traces ::  23%|##3       | 7350/31509 [00:13<00:39, 609.66it/s]","\rparsing log, completed traces ::  24%|##3       | 7415/31509 [00:13<00:38, 620.07it/s]","\rparsing log, completed traces ::  24%|##3       | 7483/31509 [00:13<00:37, 637.05it/s]","\rparsing log, completed traces ::  24%|##3       | 7555/31509 [00:14<00:36, 659.99it/s]","\rparsing log, completed traces ::  24%|##4       | 7623/31509 [00:14<00:36, 648.23it/s]","\rparsing log, completed traces ::  24%|##4       | 7689/31509 [00:14<00:36, 645.66it/s]","\rparsing log, completed traces ::  25%|##4       | 7755/31509 [00:14<00:36, 644.37it/s]","\rparsing log, completed traces ::  25%|##4       | 7820/31509 [00:14<00:37, 636.06it/s]","\rparsing log, completed traces ::  25%|##5       | 7890/31509 [00:14<00:36, 652.87it/s]","\rparsing log, completed traces ::  25%|##5       | 7956/31509 [00:14<00:36, 652.94it/s]","\rparsing log, completed traces ::  25%|##5       | 8025/31509 [00:14<00:35, 662.05it/s]","\rparsing log, completed traces ::  26%|##5       | 8093/31509 [00:14<00:35, 665.41it/s]","\rparsing log, completed traces ::  26%|##5       | 8160/31509 [00:15<00:35, 662.40it/s]","\rparsing log, completed traces ::  26%|##6       | 8232/31509 [00:15<00:34, 677.91it/s]","\rparsing log, completed traces ::  26%|##6       | 8300/31509 [00:15<00:34, 665.46it/s]","\rparsing log, completed traces ::  27%|##6       | 8367/31509 [00:15<00:34, 663.32it/s]","\rparsing log, completed traces ::  27%|##6       | 8434/31509 [00:15<00:35, 646.15it/s]","\rparsing log, completed traces ::  27%|##6       | 8503/31509 [00:15<00:34, 657.57it/s]","\rparsing log, completed traces ::  27%|##7       | 8569/31509 [00:15<00:36, 636.92it/s]","\rparsing log, completed traces ::  27%|##7       | 8635/31509 [00:15<00:35, 642.49it/s]","\rparsing log, completed traces ::  28%|##7       | 8700/31509 [00:16<01:28, 257.02it/s]","\rparsing log, completed traces ::  28%|##7       | 8766/31509 [00:16<01:12, 314.32it/s]","\rparsing log, completed traces ::  28%|##8       | 8831/31509 [00:16<01:01, 370.64it/s]","\rparsing log, completed traces ::  28%|##8       | 8893/31509 [00:16<00:54, 418.74it/s]","\rparsing log, completed traces ::  28%|##8       | 8959/31509 [00:16<00:47, 470.92it/s]","\rparsing log, completed traces ::  29%|##8       | 9020/31509 [00:16<00:45, 496.95it/s]","\rparsing log, completed traces ::  29%|##8       | 9083/31509 [00:16<00:42, 529.15it/s]","\rparsing log, completed traces ::  29%|##9       | 9147/31509 [00:17<00:40, 556.65it/s]","\rparsing log, completed traces ::  29%|##9       | 9210/31509 [00:17<00:38, 573.55it/s]","\rparsing log, completed traces ::  29%|##9       | 9272/31509 [00:17<00:38, 577.83it/s]","\rparsing log, completed traces ::  30%|##9       | 9340/31509 [00:17<00:36, 605.41it/s]","\rparsing log, completed traces ::  30%|##9       | 9403/31509 [00:17<00:36, 606.51it/s]","\rparsing log, completed traces ::  30%|###       | 9468/31509 [00:17<00:35, 618.95it/s]","\rparsing log, completed traces ::  30%|###       | 9532/31509 [00:17<00:35, 620.78it/s]","\rparsing log, completed traces ::  30%|###       | 9595/31509 [00:17<00:35, 613.92it/s]","\rparsing log, completed traces ::  31%|###       | 9660/31509 [00:17<00:35, 621.94it/s]","\rparsing log, completed traces ::  31%|###       | 9723/31509 [00:18<00:35, 614.20it/s]","\rparsing log, completed traces ::  31%|###1      | 9789/31509 [00:18<00:34, 626.00it/s]","\rparsing log, completed traces ::  31%|###1      | 9852/31509 [00:18<00:34, 624.16it/s]","\rparsing log, completed traces ::  31%|###1      | 9915/31509 [00:18<00:35, 616.65it/s]","\rparsing log, completed traces ::  32%|###1      | 9977/31509 [00:18<00:35, 613.09it/s]","\rparsing log, completed traces ::  32%|###1      | 10039/31509 [00:18<00:35, 612.31it/s]","\rparsing log, completed traces ::  32%|###2      | 10101/31509 [00:18<00:36, 593.11it/s]","\rparsing log, completed traces ::  32%|###2      | 10162/31509 [00:18<00:35, 596.28it/s]","\rparsing log, completed traces ::  32%|###2      | 10228/31509 [00:18<00:34, 612.29it/s]","\rparsing log, completed traces ::  33%|###2      | 10290/31509 [00:18<00:35, 606.00it/s]","\rparsing log, completed traces ::  33%|###2      | 10351/31509 [00:19<00:36, 572.25it/s]","\rparsing log, completed traces ::  33%|###3      | 10411/31509 [00:19<00:36, 579.99it/s]","\rparsing log, completed traces ::  33%|###3      | 10470/31509 [00:19<00:36, 579.75it/s]","\rparsing log, completed traces ::  33%|###3      | 10529/31509 [00:19<00:36, 576.14it/s]","\rparsing log, completed traces ::  34%|###3      | 10591/31509 [00:19<00:35, 588.52it/s]","\rparsing log, completed traces ::  34%|###3      | 10650/31509 [00:19<00:36, 578.19it/s]","\rparsing log, completed traces ::  34%|###3      | 10708/31509 [00:20<01:36, 214.75it/s]","\rparsing log, completed traces ::  34%|###4      | 10760/31509 [00:20<01:21, 255.53it/s]","\rparsing log, completed traces ::  34%|###4      | 10816/31509 [00:20<01:08, 304.29it/s]","\rparsing log, completed traces ::  35%|###4      | 10872/31509 [00:20<00:58, 351.89it/s]","\rparsing log, completed traces ::  35%|###4      | 10931/31509 [00:20<00:51, 401.33it/s]","\rparsing log, completed traces ::  35%|###4      | 10994/31509 [00:20<00:45, 452.89it/s]","\rparsing log, completed traces ::  35%|###5      | 11057/31509 [00:20<00:41, 494.39it/s]","\rparsing log, completed traces ::  35%|###5      | 11122/31509 [00:20<00:38, 534.64it/s]","\rparsing log, completed traces ::  35%|###5      | 11185/31509 [00:21<00:36, 558.77it/s]","\rparsing log, completed traces ::  36%|###5      | 11252/31509 [00:21<00:34, 587.03it/s]","\rparsing log, completed traces ::  36%|###5      | 11316/31509 [00:21<00:33, 600.21it/s]","\rparsing log, completed traces ::  36%|###6      | 11379/31509 [00:21<00:33, 592.56it/s]","\rparsing log, completed traces ::  36%|###6      | 11442/31509 [00:21<00:33, 602.98it/s]","\rparsing log, completed traces ::  37%|###6      | 11504/31509 [00:21<00:33, 600.53it/s]","\rparsing log, completed traces ::  37%|###6      | 11566/31509 [00:21<00:32, 605.91it/s]","\rparsing log, completed traces ::  37%|###6      | 11628/31509 [00:21<00:33, 601.07it/s]","\rparsing log, completed traces ::  37%|###7      | 11689/31509 [00:21<00:32, 600.64it/s]","\rparsing log, completed traces ::  37%|###7      | 11750/31509 [00:21<00:32, 602.66it/s]","\rparsing log, completed traces ::  37%|###7      | 11815/31509 [00:22<00:32, 612.52it/s]","\rparsing log, completed traces ::  38%|###7      | 11882/31509 [00:22<00:31, 628.32it/s]","\rparsing log, completed traces ::  38%|###7      | 11951/31509 [00:22<00:30, 645.30it/s]","\rparsing log, completed traces ::  38%|###8      | 12016/31509 [00:22<00:30, 635.71it/s]","\rparsing log, completed traces ::  38%|###8      | 12082/31509 [00:22<00:30, 638.76it/s]","\rparsing log, completed traces ::  39%|###8      | 12146/31509 [00:22<00:30, 635.38it/s]","\rparsing log, completed traces ::  39%|###8      | 12210/31509 [00:22<00:30, 634.41it/s]","\rparsing log, completed traces ::  39%|###8      | 12275/31509 [00:22<00:30, 636.30it/s]","\rparsing log, completed traces ::  39%|###9      | 12342/31509 [00:22<00:29, 645.01it/s]","\rparsing log, completed traces ::  39%|###9      | 12407/31509 [00:23<00:29, 641.32it/s]","\rparsing log, completed traces ::  40%|###9      | 12472/31509 [00:23<00:29, 643.16it/s]","\rparsing log, completed traces ::  40%|###9      | 12537/31509 [00:23<00:30, 628.91it/s]","\rparsing log, completed traces ::  40%|###9      | 12602/31509 [00:23<00:29, 631.82it/s]","\rparsing log, completed traces ::  40%|####      | 12666/31509 [00:23<00:29, 633.23it/s]","\rparsing log, completed traces ::  40%|####      | 12733/31509 [00:23<00:29, 642.90it/s]","\rparsing log, completed traces ::  41%|####      | 12798/31509 [00:23<00:29, 643.50it/s]","\rparsing log, completed traces ::  41%|####      | 12863/31509 [00:23<00:29, 624.45it/s]","\rparsing log, completed traces ::  41%|####1     | 12926/31509 [00:23<00:29, 619.93it/s]","\rparsing log, completed traces ::  41%|####1     | 12993/31509 [00:23<00:29, 634.07it/s]","\rparsing log, completed traces ::  41%|####1     | 13057/31509 [00:24<01:23, 222.27it/s]","\rparsing log, completed traces ::  42%|####1     | 13124/31509 [00:24<01:05, 279.53it/s]","\rparsing log, completed traces ::  42%|####1     | 13182/31509 [00:24<00:56, 325.10it/s]","\rparsing log, completed traces ::  42%|####2     | 13243/31509 [00:24<00:48, 375.44it/s]","\rparsing log, completed traces ::  42%|####2     | 13305/31509 [00:25<00:42, 423.91it/s]","\rparsing log, completed traces ::  42%|####2     | 13368/31509 [00:25<00:38, 467.96it/s]","\rparsing log, completed traces ::  43%|####2     | 13431/31509 [00:25<00:35, 506.96it/s]","\rparsing log, completed traces ::  43%|####2     | 13493/31509 [00:25<00:33, 533.29it/s]","\rparsing log, completed traces ::  43%|####3     | 13554/31509 [00:25<00:32, 548.52it/s]","\rparsing log, completed traces ::  43%|####3     | 13614/31509 [00:25<00:32, 555.23it/s]","\rparsing log, completed traces ::  43%|####3     | 13674/31509 [00:25<00:31, 566.15it/s]","\rparsing log, completed traces ::  44%|####3     | 13741/31509 [00:25<00:29, 593.35it/s]","\rparsing log, completed traces ::  44%|####3     | 13805/31509 [00:25<00:29, 605.84it/s]","\rparsing log, completed traces ::  44%|####4     | 13869/31509 [00:25<00:28, 613.83it/s]","\rparsing log, completed traces ::  44%|####4     | 13932/31509 [00:26<00:28, 617.66it/s]","\rparsing log, completed traces ::  44%|####4     | 13995/31509 [00:26<00:28, 620.49it/s]","\rparsing log, completed traces ::  45%|####4     | 14064/31509 [00:26<00:27, 638.56it/s]","\rparsing log, completed traces ::  45%|####4     | 14129/31509 [00:26<00:27, 637.14it/s]","\rparsing log, completed traces ::  45%|####5     | 14194/31509 [00:26<00:27, 638.33it/s]","\rparsing log, completed traces ::  45%|####5     | 14259/31509 [00:26<00:27, 636.24it/s]","\rparsing log, completed traces ::  45%|####5     | 14325/31509 [00:26<00:26, 641.49it/s]","\rparsing log, completed traces ::  46%|####5     | 14396/31509 [00:26<00:25, 661.48it/s]","\rparsing log, completed traces ::  46%|####5     | 14463/31509 [00:26<00:26, 655.53it/s]","\rparsing log, completed traces ::  46%|####6     | 14529/31509 [00:27<00:26, 643.85it/s]","\rparsing log, completed traces ::  46%|####6     | 14594/31509 [00:27<00:26, 629.19it/s]","\rparsing log, completed traces ::  47%|####6     | 14658/31509 [00:27<00:26, 629.10it/s]","\rparsing log, completed traces ::  47%|####6     | 14726/31509 [00:27<00:26, 642.47it/s]","\rparsing log, completed traces ::  47%|####6     | 14791/31509 [00:27<00:26, 642.90it/s]","\rparsing log, completed traces ::  47%|####7     | 14858/31509 [00:27<00:25, 648.37it/s]","\rparsing log, completed traces ::  47%|####7     | 14924/31509 [00:27<00:25, 650.94it/s]","\rparsing log, completed traces ::  48%|####7     | 14990/31509 [00:27<00:25, 651.65it/s]","\rparsing log, completed traces ::  48%|####7     | 15060/31509 [00:27<00:24, 664.61it/s]","\rparsing log, completed traces ::  48%|####8     | 15127/31509 [00:27<00:24, 657.17it/s]","\rparsing log, completed traces ::  48%|####8     | 15195/31509 [00:28<00:24, 662.77it/s]","\rparsing log, completed traces ::  48%|####8     | 15262/31509 [00:28<00:24, 663.31it/s]","\rparsing log, completed traces ::  49%|####8     | 15334/31509 [00:28<00:23, 679.80it/s]","\rparsing log, completed traces ::  49%|####8     | 15403/31509 [00:28<00:23, 679.43it/s]","\rparsing log, completed traces ::  49%|####9     | 15472/31509 [00:28<00:23, 681.15it/s]","\rparsing log, completed traces ::  49%|####9     | 15541/31509 [00:28<00:23, 677.27it/s]","\rparsing log, completed traces ::  50%|####9     | 15610/31509 [00:28<00:23, 680.46it/s]","\rparsing log, completed traces ::  50%|####9     | 15680/31509 [00:28<00:23, 683.77it/s]","\rparsing log, completed traces ::  50%|####9     | 15749/31509 [00:29<01:12, 216.49it/s]","\rparsing log, completed traces ::  50%|#####     | 15817/31509 [00:29<00:57, 271.17it/s]","\rparsing log, completed traces ::  50%|#####     | 15888/31509 [00:29<00:46, 333.87it/s]","\rparsing log, completed traces ::  51%|#####     | 15955/31509 [00:29<00:39, 390.64it/s]","\rparsing log, completed traces ::  51%|#####     | 16020/31509 [00:29<00:35, 440.60it/s]","\rparsing log, completed traces ::  51%|#####1    | 16091/31509 [00:30<00:30, 498.69it/s]","\rparsing log, completed traces ::  51%|#####1    | 16164/31509 [00:30<00:27, 552.93it/s]","\rparsing log, completed traces ::  52%|#####1    | 16235/31509 [00:30<00:25, 590.45it/s]","\rparsing log, completed traces ::  52%|#####1    | 16306/31509 [00:30<00:24, 621.54it/s]","\rparsing log, completed traces ::  52%|#####1    | 16375/31509 [00:30<00:24, 614.49it/s]","\rparsing log, completed traces ::  52%|#####2    | 16442/31509 [00:30<00:24, 618.90it/s]","\rparsing log, completed traces ::  52%|#####2    | 16508/31509 [00:30<00:24, 608.53it/s]","\rparsing log, completed traces ::  53%|#####2    | 16576/31509 [00:30<00:23, 627.04it/s]","\rparsing log, completed traces ::  53%|#####2    | 16645/31509 [00:30<00:23, 640.40it/s]","\rparsing log, completed traces ::  53%|#####3    | 16716/31509 [00:31<00:22, 658.76it/s]","\rparsing log, completed traces ::  53%|#####3    | 16783/31509 [00:31<00:22, 657.96it/s]","\rparsing log, completed traces ::  53%|#####3    | 16850/31509 [00:31<00:22, 654.95it/s]","\rparsing log, completed traces ::  54%|#####3    | 16919/31509 [00:31<00:21, 663.71it/s]","\rparsing log, completed traces ::  54%|#####3    | 16986/31509 [00:31<00:21, 662.96it/s]","\rparsing log, completed traces ::  54%|#####4    | 17065/31509 [00:31<00:20, 697.29it/s]","\rparsing log, completed traces ::  54%|#####4    | 17135/31509 [00:31<00:21, 678.17it/s]","\rparsing log, completed traces ::  55%|#####4    | 17204/31509 [00:31<00:21, 668.12it/s]","\rparsing log, completed traces ::  55%|#####4    | 17273/31509 [00:31<00:21, 671.22it/s]","\rparsing log, completed traces ::  55%|#####5    | 17341/31509 [00:31<00:21, 671.99it/s]","\rparsing log, completed traces ::  55%|#####5    | 17412/31509 [00:32<00:20, 682.64it/s]","\rparsing log, completed traces ::  55%|#####5    | 17481/31509 [00:32<00:20, 672.91it/s]","\rparsing log, completed traces ::  56%|#####5    | 17549/31509 [00:32<00:21, 664.45it/s]","\rparsing log, completed traces ::  56%|#####5    | 17622/31509 [00:32<00:20, 683.06it/s]","\rparsing log, completed traces ::  56%|#####6    | 17692/31509 [00:32<00:20, 685.01it/s]","\rparsing log, completed traces ::  56%|#####6    | 17762/31509 [00:32<00:19, 687.57it/s]","\rparsing log, completed traces ::  57%|#####6    | 17831/31509 [00:32<00:19, 686.44it/s]","\rparsing log, completed traces ::  57%|#####6    | 17902/31509 [00:32<00:19, 692.05it/s]","\rparsing log, completed traces ::  57%|#####7    | 17972/31509 [00:32<00:19, 687.33it/s]","\rparsing log, completed traces ::  57%|#####7    | 18045/31509 [00:32<00:19, 697.78it/s]","\rparsing log, completed traces ::  57%|#####7    | 18115/31509 [00:33<00:19, 695.69it/s]","\rparsing log, completed traces ::  58%|#####7    | 18188/31509 [00:33<00:18, 705.33it/s]","\rparsing log, completed traces ::  58%|#####7    | 18259/31509 [00:33<00:19, 693.51it/s]","\rparsing log, completed traces ::  58%|#####8    | 18329/31509 [00:33<00:19, 673.96it/s]","\rparsing log, completed traces ::  58%|#####8    | 18400/31509 [00:33<00:19, 683.62it/s]","\rparsing log, completed traces ::  59%|#####8    | 18469/31509 [00:33<00:19, 685.01it/s]","\rparsing log, completed traces ::  59%|#####8    | 18538/31509 [00:33<00:19, 667.71it/s]","\rparsing log, completed traces ::  59%|#####9    | 18605/31509 [00:33<00:19, 659.32it/s]","\rparsing log, completed traces ::  59%|#####9    | 18680/31509 [00:33<00:18, 680.90it/s]","\rparsing log, completed traces ::  60%|#####9    | 18753/31509 [00:33<00:18, 693.12it/s]","\rparsing log, completed traces ::  60%|#####9    | 18823/31509 [00:34<00:18, 670.39it/s]","\rparsing log, completed traces ::  60%|#####9    | 18891/31509 [00:35<01:01, 204.44it/s]","\rparsing log, completed traces ::  60%|######    | 18960/31509 [00:35<00:48, 258.29it/s]","\rparsing log, completed traces ::  60%|######    | 19029/31509 [00:35<00:39, 317.12it/s]","\rparsing log, completed traces ::  61%|######    | 19100/31509 [00:35<00:32, 380.48it/s]","\rparsing log, completed traces ::  61%|######    | 19171/31509 [00:35<00:27, 441.17it/s]","\rparsing log, completed traces ::  61%|######1   | 19236/31509 [00:35<00:25, 482.46it/s]","\rparsing log, completed traces ::  61%|######1   | 19305/31509 [00:35<00:23, 529.58it/s]","\rparsing log, completed traces ::  61%|######1   | 19375/31509 [00:35<00:21, 571.05it/s]","\rparsing log, completed traces ::  62%|######1   | 19447/31509 [00:35<00:19, 607.30it/s]","\rparsing log, completed traces ::  62%|######1   | 19515/31509 [00:35<00:19, 625.48it/s]","\rparsing log, completed traces ::  62%|######2   | 19584/31509 [00:36<00:18, 643.45it/s]","\rparsing log, completed traces ::  62%|######2   | 19653/31509 [00:36<00:18, 656.02it/s]","\rparsing log, completed traces ::  63%|######2   | 19722/31509 [00:36<00:18, 654.40it/s]","\rparsing log, completed traces ::  63%|######2   | 19790/31509 [00:36<00:18, 645.17it/s]","\rparsing log, completed traces ::  63%|######3   | 19856/31509 [00:36<00:18, 636.31it/s]","\rparsing log, completed traces ::  63%|######3   | 19924/31509 [00:36<00:17, 648.47it/s]","\rparsing log, completed traces ::  63%|######3   | 19990/31509 [00:36<00:17, 643.19it/s]","\rparsing log, completed traces ::  64%|######3   | 20055/31509 [00:36<00:17, 644.11it/s]","\rparsing log, completed traces ::  64%|######3   | 20123/31509 [00:36<00:17, 653.02it/s]","\rparsing log, completed traces ::  64%|######4   | 20193/31509 [00:36<00:17, 665.60it/s]","\rparsing log, completed traces ::  64%|######4   | 20264/31509 [00:37<00:16, 677.08it/s]","\rparsing log, completed traces ::  65%|######4   | 20332/31509 [00:37<00:16, 666.17it/s]","\rparsing log, completed traces ::  65%|######4   | 20407/31509 [00:37<00:16, 690.72it/s]","\rparsing log, completed traces ::  65%|######4   | 20477/31509 [00:37<00:16, 689.26it/s]","\rparsing log, completed traces ::  65%|######5   | 20547/31509 [00:37<00:16, 683.80it/s]","\rparsing log, completed traces ::  65%|######5   | 20621/31509 [00:37<00:15, 699.43it/s]","\rparsing log, completed traces ::  66%|######5   | 20695/31509 [00:37<00:15, 709.58it/s]","\rparsing log, completed traces ::  66%|######5   | 20767/31509 [00:37<00:15, 712.26it/s]","\rparsing log, completed traces ::  66%|######6   | 20839/31509 [00:37<00:15, 688.36it/s]","\rparsing log, completed traces ::  66%|######6   | 20909/31509 [00:37<00:15, 687.01it/s]","\rparsing log, completed traces ::  67%|######6   | 20978/31509 [00:38<00:15, 669.54it/s]","\rparsing log, completed traces ::  67%|######6   | 21046/31509 [00:38<00:15, 661.25it/s]","\rparsing log, completed traces ::  67%|######7   | 21113/31509 [00:38<00:15, 661.38it/s]","\rparsing log, completed traces ::  67%|######7   | 21180/31509 [00:38<00:15, 654.86it/s]","\rparsing log, completed traces ::  67%|######7   | 21253/31509 [00:38<00:15, 675.57it/s]","\rparsing log, completed traces ::  68%|######7   | 21322/31509 [00:38<00:14, 679.65it/s]","\rparsing log, completed traces ::  68%|######7   | 21391/31509 [00:38<00:15, 665.24it/s]","\rparsing log, completed traces ::  68%|######8   | 21466/31509 [00:38<00:14, 689.90it/s]","\rparsing log, completed traces ::  68%|######8   | 21537/31509 [00:38<00:14, 694.86it/s]","\rparsing log, completed traces ::  69%|######8   | 21607/31509 [00:39<00:14, 688.87it/s]","\rparsing log, completed traces ::  69%|######8   | 21676/31509 [00:39<00:14, 671.09it/s]","\rparsing log, completed traces ::  69%|######9   | 21754/31509 [00:39<00:13, 701.19it/s]","\rparsing log, completed traces ::  69%|######9   | 21825/31509 [00:39<00:13, 697.40it/s]","\rparsing log, completed traces ::  70%|######9   | 21901/31509 [00:39<00:13, 712.27it/s]","\rparsing log, completed traces ::  70%|######9   | 21973/31509 [00:39<00:13, 699.92it/s]","\rparsing log, completed traces ::  70%|######9   | 22049/31509 [00:39<00:13, 717.04it/s]","\rparsing log, completed traces ::  70%|#######   | 22121/31509 [00:39<00:13, 707.13it/s]","\rparsing log, completed traces ::  70%|#######   | 22194/31509 [00:39<00:13, 709.95it/s]","\rparsing log, completed traces ::  71%|#######   | 22266/31509 [00:39<00:13, 698.36it/s]","\rparsing log, completed traces ::  71%|#######   | 22336/31509 [00:40<00:13, 694.63it/s]","\rparsing log, completed traces ::  71%|#######1  | 22406/31509 [00:40<00:13, 688.98it/s]","\rparsing log, completed traces ::  71%|#######1  | 22481/31509 [00:40<00:12, 706.42it/s]","\rparsing log, completed traces ::  72%|#######1  | 22552/31509 [00:41<00:46, 193.44it/s]","\rparsing log, completed traces ::  72%|#######1  | 22623/31509 [00:41<00:36, 246.03it/s]","\rparsing log, completed traces ::  72%|#######2  | 22691/31509 [00:41<00:29, 301.49it/s]","\rparsing log, completed traces ::  72%|#######2  | 22763/31509 [00:41<00:23, 366.05it/s]","\rparsing log, completed traces ::  72%|#######2  | 22834/31509 [00:41<00:20, 427.64it/s]","\rparsing log, completed traces ::  73%|#######2  | 22900/31509 [00:41<00:18, 474.97it/s]","\rparsing log, completed traces ::  73%|#######2  | 22966/31509 [00:41<00:16, 510.97it/s]","\rparsing log, completed traces ::  73%|#######3  | 23035/31509 [00:41<00:15, 553.83it/s]","\rparsing log, completed traces ::  73%|#######3  | 23102/31509 [00:42<00:14, 583.63it/s]","\rparsing log, completed traces ::  74%|#######3  | 23169/31509 [00:42<00:13, 598.56it/s]","\rparsing log, completed traces ::  74%|#######3  | 23236/31509 [00:42<00:13, 617.41it/s]","\rparsing log, completed traces ::  74%|#######3  | 23302/31509 [00:42<00:13, 623.05it/s]","\rparsing log, completed traces ::  74%|#######4  | 23368/31509 [00:42<00:13, 625.63it/s]","\rparsing log, completed traces ::  74%|#######4  | 23436/31509 [00:42<00:12, 639.07it/s]","\rparsing log, completed traces ::  75%|#######4  | 23502/31509 [00:42<00:12, 641.55it/s]","\rparsing log, completed traces ::  75%|#######4  | 23573/31509 [00:42<00:12, 659.88it/s]","\rparsing log, completed traces ::  75%|#######5  | 23641/31509 [00:42<00:11, 664.87it/s]","\rparsing log, completed traces ::  75%|#######5  | 23710/31509 [00:42<00:11, 670.83it/s]","\rparsing log, completed traces ::  75%|#######5  | 23778/31509 [00:43<00:11, 670.16it/s]","\rparsing log, completed traces ::  76%|#######5  | 23846/31509 [00:43<00:11, 671.61it/s]","\rparsing log, completed traces ::  76%|#######5  | 23915/31509 [00:43<00:11, 674.97it/s]","\rparsing log, completed traces ::  76%|#######6  | 23983/31509 [00:43<00:11, 670.86it/s]","\rparsing log, completed traces ::  76%|#######6  | 24051/31509 [00:43<00:11, 664.14it/s]","\rparsing log, completed traces ::  77%|#######6  | 24118/31509 [00:43<00:11, 663.64it/s]","\rparsing log, completed traces ::  77%|#######6  | 24185/31509 [00:43<00:11, 662.59it/s]","\rparsing log, completed traces ::  77%|#######6  | 24252/31509 [00:43<00:10, 663.92it/s]","\rparsing log, completed traces ::  77%|#######7  | 24321/31509 [00:43<00:10, 670.33it/s]","\rparsing log, completed traces ::  77%|#######7  | 24392/31509 [00:43<00:10, 679.33it/s]","\rparsing log, completed traces ::  78%|#######7  | 24460/31509 [00:44<00:10, 663.15it/s]","\rparsing log, completed traces ::  78%|#######7  | 24531/31509 [00:44<00:10, 675.16it/s]","\rparsing log, completed traces ::  78%|#######8  | 24605/31509 [00:44<00:09, 692.88it/s]","\rparsing log, completed traces ::  78%|#######8  | 24675/31509 [00:44<00:09, 692.87it/s]","\rparsing log, completed traces ::  79%|#######8  | 24745/31509 [00:44<00:09, 690.51it/s]","\rparsing log, completed traces ::  79%|#######8  | 24815/31509 [00:44<00:09, 686.85it/s]","\rparsing log, completed traces ::  79%|#######8  | 24890/31509 [00:44<00:09, 703.76it/s]","\rparsing log, completed traces ::  79%|#######9  | 24961/31509 [00:44<00:09, 693.78it/s]","\rparsing log, completed traces ::  79%|#######9  | 25033/31509 [00:44<00:09, 697.31it/s]","\rparsing log, completed traces ::  80%|#######9  | 25103/31509 [00:45<00:09, 672.63it/s]","\rparsing log, completed traces ::  80%|#######9  | 25177/31509 [00:45<00:09, 690.33it/s]","\rparsing log, completed traces ::  80%|########  | 25247/31509 [00:45<00:09, 676.90it/s]","\rparsing log, completed traces ::  80%|########  | 25315/31509 [00:45<00:09, 650.26it/s]","\rparsing log, completed traces ::  81%|########  | 25388/31509 [00:45<00:09, 670.77it/s]","\rparsing log, completed traces ::  81%|########  | 25456/31509 [00:45<00:08, 672.72it/s]","\rparsing log, completed traces ::  81%|########1 | 25524/31509 [00:45<00:08, 674.27it/s]","\rparsing log, completed traces ::  81%|########1 | 25597/31509 [00:45<00:08, 689.23it/s]","\rparsing log, completed traces ::  81%|########1 | 25667/31509 [00:45<00:08, 674.37it/s]","\rparsing log, completed traces ::  82%|########1 | 25735/31509 [00:45<00:08, 666.95it/s]","\rparsing log, completed traces ::  82%|########1 | 25802/31509 [00:46<00:08, 661.31it/s]","\rparsing log, completed traces ::  82%|########2 | 25869/31509 [00:46<00:08, 652.80it/s]","\rparsing log, completed traces ::  82%|########2 | 25939/31509 [00:46<00:08, 665.66it/s]","\rparsing log, completed traces ::  83%|########2 | 26006/31509 [00:46<00:08, 647.32it/s]","\rparsing log, completed traces ::  83%|########2 | 26071/31509 [00:46<00:08, 645.94it/s]","\rparsing log, completed traces ::  83%|########2 | 26138/31509 [00:46<00:08, 652.75it/s]","\rparsing log, completed traces ::  83%|########3 | 26205/31509 [00:46<00:08, 656.43it/s]","\rparsing log, completed traces ::  83%|########3 | 26272/31509 [00:46<00:07, 660.36it/s]","\rparsing log, completed traces ::  84%|########3 | 26343/31509 [00:46<00:07, 673.16it/s]","\rparsing log, completed traces ::  84%|########3 | 26412/31509 [00:47<00:07, 673.25it/s]","\rparsing log, completed traces ::  84%|########4 | 26480/31509 [00:48<00:29, 171.33it/s]","\rparsing log, completed traces ::  84%|########4 | 26540/31509 [00:48<00:23, 212.58it/s]","\rparsing log, completed traces ::  84%|########4 | 26604/31509 [00:48<00:18, 264.48it/s]","\rparsing log, completed traces ::  85%|########4 | 26671/31509 [00:48<00:14, 324.22it/s]","\rparsing log, completed traces ::  85%|########4 | 26742/31509 [00:48<00:12, 390.79it/s]","\rparsing log, completed traces ::  85%|########5 | 26807/31509 [00:48<00:10, 440.94it/s]","\rparsing log, completed traces ::  85%|########5 | 26872/31509 [00:48<00:09, 485.41it/s]","\rparsing log, completed traces ::  85%|########5 | 26938/31509 [00:48<00:08, 526.12it/s]","\rparsing log, completed traces ::  86%|########5 | 27002/31509 [00:48<00:08, 551.01it/s]","\rparsing log, completed traces ::  86%|########5 | 27066/31509 [00:49<00:07, 567.97it/s]","\rparsing log, completed traces ::  86%|########6 | 27137/31509 [00:49<00:07, 602.84it/s]","\rparsing log, completed traces ::  86%|########6 | 27206/31509 [00:49<00:06, 626.06it/s]","\rparsing log, completed traces ::  87%|########6 | 27275/31509 [00:49<00:06, 644.19it/s]","\rparsing log, completed traces ::  87%|########6 | 27342/31509 [00:49<00:06, 631.80it/s]","\rparsing log, completed traces ::  87%|########6 | 27409/31509 [00:49<00:06, 642.07it/s]","\rparsing log, completed traces ::  87%|########7 | 27477/31509 [00:49<00:06, 651.82it/s]","\rparsing log, completed traces ::  87%|########7 | 27544/31509 [00:49<00:06, 649.82it/s]","\rparsing log, completed traces ::  88%|########7 | 27615/31509 [00:49<00:05, 665.23it/s]","\rparsing log, completed traces ::  88%|########7 | 27686/31509 [00:49<00:05, 674.72it/s]","\rparsing log, completed traces ::  88%|########8 | 27759/31509 [00:50<00:05, 690.64it/s]","\rparsing log, completed traces ::  88%|########8 | 27829/31509 [00:50<00:05, 677.22it/s]","\rparsing log, completed traces ::  89%|########8 | 27897/31509 [00:50<00:05, 664.84it/s]","\rparsing log, completed traces ::  89%|########8 | 27964/31509 [00:50<00:05, 654.48it/s]","\rparsing log, completed traces ::  89%|########8 | 28033/31509 [00:50<00:05, 663.45it/s]","\rparsing log, completed traces ::  89%|########9 | 28100/31509 [00:50<00:05, 650.03it/s]","\rparsing log, completed traces ::  89%|########9 | 28166/31509 [00:50<00:05, 642.94it/s]","\rparsing log, completed traces ::  90%|########9 | 28236/31509 [00:50<00:04, 657.29it/s]","\rparsing log, completed traces ::  90%|########9 | 28302/31509 [00:50<00:04, 653.07it/s]","\rparsing log, completed traces ::  90%|######### | 28371/31509 [00:50<00:04, 662.42it/s]","\rparsing log, completed traces ::  90%|######### | 28438/31509 [00:51<00:04, 647.08it/s]","\rparsing log, completed traces ::  90%|######### | 28506/31509 [00:51<00:04, 655.63it/s]","\rparsing log, completed traces ::  91%|######### | 28577/31509 [00:51<00:04, 668.38it/s]","\rparsing log, completed traces ::  91%|######### | 28644/31509 [00:51<00:04, 664.80it/s]","\rparsing log, completed traces ::  91%|#########1| 28711/31509 [00:51<00:04, 645.11it/s]","\rparsing log, completed traces ::  91%|#########1| 28776/31509 [00:51<00:04, 624.96it/s]","\rparsing log, completed traces ::  92%|#########1| 28839/31509 [00:51<00:04, 615.16it/s]","\rparsing log, completed traces ::  92%|#########1| 28901/31509 [00:51<00:04, 610.46it/s]","\rparsing log, completed traces ::  92%|#########1| 28969/31509 [00:51<00:04, 629.33it/s]","\rparsing log, completed traces ::  92%|#########2| 29035/31509 [00:52<00:03, 634.65it/s]","\rparsing log, completed traces ::  92%|#########2| 29099/31509 [00:52<00:03, 629.96it/s]","\rparsing log, completed traces ::  93%|#########2| 29165/31509 [00:52<00:03, 636.21it/s]","\rparsing log, completed traces ::  93%|#########2| 29234/31509 [00:52<00:03, 650.47it/s]","\rparsing log, completed traces ::  93%|#########2| 29300/31509 [00:52<00:03, 651.80it/s]","\rparsing log, completed traces ::  93%|#########3| 29366/31509 [00:52<00:03, 641.16it/s]","\rparsing log, completed traces ::  93%|#########3| 29431/31509 [00:52<00:03, 630.89it/s]","\rparsing log, completed traces ::  94%|#########3| 29495/31509 [00:52<00:03, 615.78it/s]","\rparsing log, completed traces ::  94%|#########3| 29557/31509 [00:52<00:03, 589.59it/s]","\rparsing log, completed traces ::  94%|#########4| 29631/31509 [00:52<00:02, 628.03it/s]","\rparsing log, completed traces ::  94%|#########4| 29698/31509 [00:53<00:02, 638.41it/s]","\rparsing log, completed traces ::  94%|#########4| 29763/31509 [00:53<00:02, 638.44it/s]","\rparsing log, completed traces ::  95%|#########4| 29828/31509 [00:53<00:02, 632.16it/s]","\rparsing log, completed traces ::  95%|#########4| 29895/31509 [00:53<00:02, 642.23it/s]","\rparsing log, completed traces ::  95%|#########5| 29960/31509 [00:53<00:02, 643.36it/s]","\rparsing log, completed traces ::  95%|#########5| 30026/31509 [00:53<00:02, 644.07it/s]","\rparsing log, completed traces ::  96%|#########5| 30092/31509 [00:53<00:02, 647.25it/s]","\rparsing log, completed traces ::  96%|#########5| 30162/31509 [00:53<00:02, 659.19it/s]","\rparsing log, completed traces ::  96%|#########5| 30229/31509 [00:53<00:01, 661.56it/s]","\rparsing log, completed traces ::  96%|#########6| 30296/31509 [00:53<00:01, 660.13it/s]","\rparsing log, completed traces ::  96%|#########6| 30367/31509 [00:54<00:01, 673.87it/s]","\rparsing log, completed traces ::  97%|#########6| 30435/31509 [00:54<00:01, 668.27it/s]","\rparsing log, completed traces ::  97%|#########6| 30502/31509 [00:54<00:01, 655.30it/s]","\rparsing log, completed traces ::  97%|#########7| 30576/31509 [00:54<00:01, 677.85it/s]","\rparsing log, completed traces ::  97%|#########7| 30644/31509 [00:54<00:01, 673.37it/s]","\rparsing log, completed traces ::  97%|#########7| 30713/31509 [00:54<00:01, 675.68it/s]","\rparsing log, completed traces ::  98%|#########7| 30781/31509 [00:54<00:01, 673.46it/s]","\rparsing log, completed traces ::  98%|#########7| 30849/31509 [00:55<00:04, 156.10it/s]","\rparsing log, completed traces ::  98%|#########8| 30913/31509 [00:56<00:02, 199.11it/s]","\rparsing log, completed traces ::  98%|#########8| 30980/31509 [00:56<00:02, 251.55it/s]","\rparsing log, completed traces ::  99%|#########8| 31049/31509 [00:56<00:01, 311.54it/s]","\rparsing log, completed traces ::  99%|#########8| 31122/31509 [00:56<00:01, 379.17it/s]","\rparsing log, completed traces ::  99%|#########8| 31193/31509 [00:56<00:00, 440.74it/s]","\rparsing log, completed traces ::  99%|#########9| 31259/31509 [00:56<00:00, 484.99it/s]","\rparsing log, completed traces ::  99%|#########9| 31325/31509 [00:56<00:00, 504.76it/s]","\rparsing log, completed traces :: 100%|#########9| 31388/31509 [00:56<00:00, 533.83it/s]","\rparsing log, completed traces :: 100%|#########9| 31455/31509 [00:56<00:00, 567.43it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:56<00:00, 553.34it/s]","\n","[data] Loading XES: /workspace/data/Road_Traffic_Fine_Management_Process.xes","\n","\rparsing log, completed traces ::   0%|          | 0/150370 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 744/150370 [00:00<00:20, 7434.14it/s]","\rparsing log, completed traces ::   1%|1         | 1547/150370 [00:00<00:19, 7783.99it/s]","\rparsing log, completed traces ::   2%|1         | 2326/150370 [00:00<00:40, 3632.61it/s]","\rparsing log, completed traces ::   2%|2         | 3101/150370 [00:00<00:32, 4589.01it/s]","\rparsing log, completed traces ::   3%|2         | 3869/150370 [00:00<00:27, 5360.55it/s]","\rparsing log, completed traces ::   3%|3         | 4648/150370 [00:00<00:24, 5992.63it/s]","\rparsing log, completed traces ::   4%|3         | 5433/150370 [00:00<00:22, 6495.76it/s]","\rparsing log, completed traces ::   4%|4         | 6215/150370 [00:01<00:20, 6865.93it/s]","\rparsing log, completed traces ::   5%|4         | 6998/150370 [00:01<00:20, 7121.88it/s]","\rparsing log, completed traces ::   5%|5         | 7780/150370 [00:01<00:19, 7322.07it/s]","\rparsing log, completed traces ::   6%|5         | 8610/150370 [00:01<00:18, 7606.23it/s]","\rparsing log, completed traces ::   6%|6         | 9432/150370 [00:01<00:18, 7783.59it/s]","\rparsing log, completed traces ::   7%|6         | 10233/150370 [00:01<00:17, 7849.37it/s]","\rparsing log, completed traces ::   7%|7         | 11030/150370 [00:01<00:17, 7845.37it/s]","\rparsing log, completed traces ::   8%|7         | 11823/150370 [00:01<00:17, 7837.54it/s]","\rparsing log, completed traces ::   8%|8         | 12613/150370 [00:01<00:17, 7817.55it/s]","\rparsing log, completed traces ::   9%|8         | 13399/150370 [00:02<00:32, 4159.63it/s]","\rparsing log, completed traces ::   9%|9         | 14171/150370 [00:02<00:28, 4814.03it/s]","\rparsing log, completed traces ::  10%|9         | 14966/150370 [00:02<00:24, 5464.32it/s]","\rparsing log, completed traces ::  10%|#         | 15779/150370 [00:02<00:22, 6074.63it/s]","\rparsing log, completed traces ::  11%|#1        | 16610/150370 [00:02<00:20, 6625.32it/s]","\rparsing log, completed traces ::  12%|#1        | 17437/150370 [00:02<00:18, 7053.25it/s]","\rparsing log, completed traces ::  12%|#2        | 18238/150370 [00:02<00:18, 7311.35it/s]","\rparsing log, completed traces ::  13%|#2        | 19054/150370 [00:02<00:17, 7547.18it/s]","\rparsing log, completed traces ::  13%|#3        | 19856/150370 [00:03<00:17, 7660.10it/s]","\rparsing log, completed traces ::  14%|#3        | 20659/150370 [00:03<00:16, 7761.85it/s]","\rparsing log, completed traces ::  14%|#4        | 21537/150370 [00:03<00:15, 8058.19it/s]","\rparsing log, completed traces ::  15%|#4        | 22395/150370 [00:03<00:15, 8210.38it/s]","\rparsing log, completed traces ::  15%|#5        | 23229/150370 [00:03<00:15, 8248.35it/s]","\rparsing log, completed traces ::  16%|#6        | 24062/150370 [00:03<00:15, 7931.03it/s]","\rparsing log, completed traces ::  17%|#6        | 24864/150370 [00:03<00:16, 7407.88it/s]","\rparsing log, completed traces ::  17%|#7        | 25617/150370 [00:03<00:17, 7166.71it/s]","\rparsing log, completed traces ::  18%|#7        | 26343/150370 [00:04<00:35, 3527.42it/s]","\rparsing log, completed traces ::  18%|#7        | 26970/150370 [00:04<00:31, 3971.10it/s]","\rparsing log, completed traces ::  18%|#8        | 27605/150370 [00:04<00:27, 4418.70it/s]","\rparsing log, completed traces ::  19%|#8        | 28282/150370 [00:04<00:24, 4915.94it/s]","\rparsing log, completed traces ::  19%|#9        | 29049/150370 [00:04<00:21, 5555.27it/s]","\rparsing log, completed traces ::  20%|#9        | 29813/150370 [00:04<00:19, 6056.64it/s]","\rparsing log, completed traces ::  20%|##        | 30504/150370 [00:04<00:19, 6229.87it/s]","\rparsing log, completed traces ::  21%|##        | 31189/150370 [00:04<00:18, 6343.10it/s]","\rparsing log, completed traces ::  21%|##1       | 31868/150370 [00:05<00:18, 6436.19it/s]","\rparsing log, completed traces ::  22%|##1       | 32543/150370 [00:05<00:18, 6509.11it/s]","\rparsing log, completed traces ::  22%|##2       | 33217/150370 [00:05<00:17, 6571.24it/s]","\rparsing log, completed traces ::  23%|##2       | 33903/150370 [00:05<00:17, 6651.36it/s]","\rparsing log, completed traces ::  23%|##2       | 34580/150370 [00:05<00:17, 6673.63it/s]","\rparsing log, completed traces ::  23%|##3       | 35256/150370 [00:05<00:17, 6680.00it/s]","\rparsing log, completed traces ::  24%|##3       | 35932/150370 [00:05<00:17, 6701.31it/s]","\rparsing log, completed traces ::  24%|##4       | 36617/150370 [00:05<00:16, 6736.86it/s]","\rparsing log, completed traces ::  25%|##4       | 37307/150370 [00:05<00:16, 6765.48it/s]","\rparsing log, completed traces ::  25%|##5       | 38016/150370 [00:06<00:16, 6860.90it/s]","\rparsing log, completed traces ::  26%|##5       | 38704/150370 [00:06<00:35, 3121.71it/s]","\rparsing log, completed traces ::  26%|##6       | 39420/150370 [00:06<00:29, 3776.36it/s]","\rparsing log, completed traces ::  27%|##6       | 40126/150370 [00:06<00:25, 4393.59it/s]","\rparsing log, completed traces ::  27%|##7       | 40840/150370 [00:06<00:22, 4974.81it/s]","\rparsing log, completed traces ::  28%|##7       | 41587/150370 [00:06<00:19, 5553.62it/s]","\rparsing log, completed traces ::  28%|##8       | 42324/150370 [00:07<00:18, 6002.48it/s]","\rparsing log, completed traces ::  29%|##8       | 43046/150370 [00:07<00:17, 6306.31it/s]","\rparsing log, completed traces ::  29%|##9       | 43787/150370 [00:07<00:16, 6606.52it/s]","\rparsing log, completed traces ::  30%|##9       | 44529/150370 [00:07<00:15, 6818.73it/s]","\rparsing log, completed traces ::  30%|###       | 45249/150370 [00:07<00:15, 6900.06it/s]","\rparsing log, completed traces ::  31%|###       | 45970/150370 [00:07<00:14, 6982.17it/s]","\rparsing log, completed traces ::  31%|###1      | 46688/150370 [00:07<00:14, 7004.56it/s]","\rparsing log, completed traces ::  32%|###1      | 47402/150370 [00:07<00:14, 6999.33it/s]","\rparsing log, completed traces ::  32%|###2      | 48139/150370 [00:07<00:14, 7106.80it/s]","\rparsing log, completed traces ::  33%|###2      | 48946/150370 [00:07<00:13, 7382.05it/s]","\rparsing log, completed traces ::  33%|###3      | 49690/150370 [00:08<00:13, 7321.48it/s]","\rparsing log, completed traces ::  34%|###3      | 50426/150370 [00:08<00:13, 7163.57it/s]","\rparsing log, completed traces ::  34%|###4      | 51146/150370 [00:08<00:13, 7155.28it/s]","\rparsing log, completed traces ::  34%|###4      | 51864/150370 [00:08<00:13, 7147.26it/s]","\rparsing log, completed traces ::  35%|###4      | 52581/150370 [00:08<00:13, 7105.43it/s]","\rparsing log, completed traces ::  35%|###5      | 53293/150370 [00:08<00:13, 7104.14it/s]","\rparsing log, completed traces ::  36%|###5      | 54005/150370 [00:08<00:13, 7107.28it/s]","\rparsing log, completed traces ::  36%|###6      | 54717/150370 [00:09<00:32, 2911.63it/s]","\rparsing log, completed traces ::  37%|###6      | 55379/150370 [00:09<00:27, 3460.93it/s]","\rparsing log, completed traces ::  37%|###7      | 56130/150370 [00:09<00:22, 4164.64it/s]","\rparsing log, completed traces ::  38%|###7      | 56912/150370 [00:09<00:19, 4893.51it/s]","\rparsing log, completed traces ::  38%|###8      | 57677/150370 [00:09<00:16, 5504.23it/s]","\rparsing log, completed traces ::  39%|###8      | 58421/150370 [00:09<00:15, 5969.80it/s]","\rparsing log, completed traces ::  39%|###9      | 59160/150370 [00:09<00:14, 6321.43it/s]","\rparsing log, completed traces ::  40%|###9      | 59958/150370 [00:09<00:13, 6764.44it/s]","\rparsing log, completed traces ::  40%|####      | 60725/150370 [00:10<00:12, 7013.18it/s]","\rparsing log, completed traces ::  41%|####      | 61577/150370 [00:10<00:11, 7421.83it/s]","\rparsing log, completed traces ::  42%|####1     | 62418/150370 [00:10<00:11, 7705.17it/s]","\rparsing log, completed traces ::  42%|####2     | 63216/150370 [00:10<00:11, 7681.71it/s]","\rparsing log, completed traces ::  43%|####2     | 64003/150370 [00:10<00:11, 7662.73it/s]","\rparsing log, completed traces ::  43%|####3     | 64784/150370 [00:10<00:11, 7705.26it/s]","\rparsing log, completed traces ::  44%|####3     | 65564/150370 [00:10<00:10, 7732.63it/s]","\rparsing log, completed traces ::  44%|####4     | 66363/150370 [00:10<00:10, 7808.13it/s]","\rparsing log, completed traces ::  45%|####4     | 67153/150370 [00:10<00:10, 7810.09it/s]","\rparsing log, completed traces ::  45%|####5     | 68039/150370 [00:10<00:10, 8122.06it/s]","\rparsing log, completed traces ::  46%|####5     | 68854/150370 [00:11<00:10, 8101.75it/s]","\rparsing log, completed traces ::  46%|####6     | 69666/150370 [00:11<00:10, 7966.59it/s]","\rparsing log, completed traces ::  47%|####6     | 70465/150370 [00:11<00:10, 7872.90it/s]","\rparsing log, completed traces ::  47%|####7     | 71254/150370 [00:11<00:10, 7807.00it/s]","\rparsing log, completed traces ::  48%|####7     | 72036/150370 [00:11<00:10, 7727.31it/s]","\rparsing log, completed traces ::  48%|####8     | 72810/150370 [00:11<00:10, 7692.14it/s]","\rparsing log, completed traces ::  49%|####8     | 73580/150370 [00:12<00:25, 2982.99it/s]","\rparsing log, completed traces ::  49%|####9     | 74358/150370 [00:12<00:20, 3653.67it/s]","\rparsing log, completed traces ::  50%|#####     | 75214/150370 [00:12<00:16, 4465.74it/s]","\rparsing log, completed traces ::  51%|#####     | 75995/150370 [00:12<00:14, 5107.81it/s]","\rparsing log, completed traces ::  51%|#####1    | 76817/150370 [00:12<00:12, 5777.43it/s]","\rparsing log, completed traces ::  52%|#####1    | 77571/150370 [00:12<00:11, 6177.88it/s]","\rparsing log, completed traces ::  52%|#####2    | 78323/150370 [00:12<00:11, 6500.97it/s]","\rparsing log, completed traces ::  53%|#####2    | 79073/150370 [00:12<00:10, 6749.06it/s]","\rparsing log, completed traces ::  53%|#####3    | 79821/150370 [00:12<00:10, 6919.50it/s]","\rparsing log, completed traces ::  54%|#####3    | 80566/150370 [00:13<00:09, 7042.28it/s]","\rparsing log, completed traces ::  54%|#####4    | 81308/150370 [00:13<00:09, 7118.87it/s]","\rparsing log, completed traces ::  55%|#####4    | 82074/150370 [00:13<00:09, 7274.86it/s]","\rparsing log, completed traces ::  55%|#####5    | 82903/150370 [00:13<00:08, 7548.86it/s]","\rparsing log, completed traces ::  56%|#####5    | 83672/150370 [00:13<00:08, 7546.52it/s]","\rparsing log, completed traces ::  56%|#####6    | 84437/150370 [00:13<00:08, 7505.97it/s]","\rparsing log, completed traces ::  57%|#####6    | 85195/150370 [00:13<00:08, 7392.77it/s]","\rparsing log, completed traces ::  57%|#####7    | 85940/150370 [00:13<00:08, 7372.24it/s]","\rparsing log, completed traces ::  58%|#####7    | 86681/150370 [00:13<00:08, 7372.26it/s]","\rparsing log, completed traces ::  58%|#####8    | 87443/150370 [00:14<00:08, 7443.41it/s]","\rparsing log, completed traces ::  59%|#####8    | 88190/150370 [00:14<00:08, 7445.56it/s]","\rparsing log, completed traces ::  59%|#####9    | 88943/150370 [00:14<00:08, 7467.90it/s]","\rparsing log, completed traces ::  60%|#####9    | 89740/150370 [00:14<00:07, 7615.46it/s]","\rparsing log, completed traces ::  60%|######    | 90503/150370 [00:14<00:07, 7613.70it/s]","\rparsing log, completed traces ::  61%|######    | 91265/150370 [00:14<00:07, 7592.99it/s]","\rparsing log, completed traces ::  61%|######1   | 92040/150370 [00:14<00:07, 7631.81it/s]","\rparsing log, completed traces ::  62%|######1   | 92805/150370 [00:14<00:07, 7635.69it/s]","\rparsing log, completed traces ::  62%|######2   | 93581/150370 [00:14<00:07, 7672.65it/s]","\rparsing log, completed traces ::  63%|######2   | 94365/150370 [00:14<00:07, 7720.56it/s]","\rparsing log, completed traces ::  63%|######3   | 95166/150370 [00:15<00:07, 7804.91it/s]","\rparsing log, completed traces ::  64%|######3   | 95954/150370 [00:15<00:06, 7822.45it/s]","\rparsing log, completed traces ::  64%|######4   | 96737/150370 [00:15<00:19, 2719.60it/s]","\rparsing log, completed traces ::  65%|######4   | 97486/150370 [00:15<00:15, 3337.24it/s]","\rparsing log, completed traces ::  65%|######5   | 98251/150370 [00:16<00:12, 4010.46it/s]","\rparsing log, completed traces ::  66%|######5   | 99028/150370 [00:16<00:10, 4695.81it/s]","\rparsing log, completed traces ::  66%|######6   | 99778/150370 [00:16<00:09, 5264.22it/s]","\rparsing log, completed traces ::  67%|######6   | 100559/150370 [00:16<00:08, 5842.65it/s]","\rparsing log, completed traces ::  67%|######7   | 101332/150370 [00:16<00:07, 6304.90it/s]","\rparsing log, completed traces ::  68%|######7   | 102114/150370 [00:16<00:07, 6697.82it/s]","\rparsing log, completed traces ::  68%|######8   | 102877/150370 [00:16<00:06, 6949.84it/s]","\rparsing log, completed traces ::  69%|######8   | 103636/150370 [00:16<00:06, 7119.53it/s]","\rparsing log, completed traces ::  69%|######9   | 104394/150370 [00:16<00:06, 7201.12it/s]","\rparsing log, completed traces ::  70%|######9   | 105147/150370 [00:16<00:06, 7252.12it/s]","\rparsing log, completed traces ::  70%|#######   | 105895/150370 [00:17<00:06, 7299.97it/s]","\rparsing log, completed traces ::  71%|#######   | 106642/150370 [00:17<00:05, 7317.19it/s]","\rparsing log, completed traces ::  71%|#######1  | 107386/150370 [00:17<00:05, 7352.16it/s]","\rparsing log, completed traces ::  72%|#######1  | 108130/150370 [00:17<00:05, 7349.29it/s]","\rparsing log, completed traces ::  72%|#######2  | 108871/150370 [00:17<00:05, 7362.82it/s]","\rparsing log, completed traces ::  73%|#######2  | 109612/150370 [00:17<00:05, 7370.71it/s]","\rparsing log, completed traces ::  73%|#######3  | 110352/150370 [00:17<00:05, 7335.99it/s]","\rparsing log, completed traces ::  74%|#######3  | 111088/150370 [00:17<00:05, 7333.53it/s]","\rparsing log, completed traces ::  74%|#######4  | 111837/150370 [00:17<00:05, 7370.30it/s]","\rparsing log, completed traces ::  75%|#######4  | 112575/150370 [00:17<00:05, 7308.82it/s]","\rparsing log, completed traces ::  75%|#######5  | 113307/150370 [00:18<00:05, 7252.41it/s]","\rparsing log, completed traces ::  76%|#######5  | 114186/150370 [00:18<00:04, 7706.20it/s]","\rparsing log, completed traces ::  77%|#######6  | 115110/150370 [00:18<00:04, 8161.02it/s]","\rparsing log, completed traces ::  77%|#######7  | 116001/150370 [00:18<00:04, 8383.38it/s]","\rparsing log, completed traces ::  78%|#######7  | 116972/150370 [00:18<00:03, 8779.19it/s]","\rparsing log, completed traces ::  78%|#######8  | 117851/150370 [00:18<00:03, 8620.69it/s]","\rparsing log, completed traces ::  79%|#######8  | 118715/150370 [00:18<00:03, 8044.00it/s]","\rparsing log, completed traces ::  79%|#######9  | 119528/150370 [00:18<00:04, 7664.73it/s]","\rparsing log, completed traces ::  80%|########  | 120303/150370 [00:18<00:04, 7441.13it/s]","\rparsing log, completed traces ::  81%|########  | 121053/150370 [00:19<00:04, 7302.89it/s]","\rparsing log, completed traces ::  81%|########  | 121787/150370 [00:19<00:03, 7187.27it/s]","\rparsing log, completed traces ::  81%|########1 | 122508/150370 [00:19<00:11, 2433.67it/s]","\rparsing log, completed traces ::  82%|########1 | 123218/150370 [00:20<00:09, 2992.32it/s]","\rparsing log, completed traces ::  82%|########2 | 123915/150370 [00:20<00:07, 3573.15it/s]","\rparsing log, completed traces ::  83%|########2 | 124598/150370 [00:20<00:06, 4132.01it/s]","\rparsing log, completed traces ::  83%|########3 | 125308/150370 [00:20<00:05, 4718.77it/s]","\rparsing log, completed traces ::  84%|########3 | 126005/150370 [00:20<00:04, 5213.93it/s]","\rparsing log, completed traces ::  84%|########4 | 126734/150370 [00:20<00:04, 5709.99it/s]","\rparsing log, completed traces ::  85%|########4 | 127454/150370 [00:20<00:03, 6087.41it/s]","\rparsing log, completed traces ::  85%|########5 | 128169/150370 [00:20<00:03, 6369.96it/s]","\rparsing log, completed traces ::  86%|########5 | 128898/150370 [00:20<00:03, 6623.13it/s]","\rparsing log, completed traces ::  86%|########6 | 129642/150370 [00:20<00:03, 6834.82it/s]","\rparsing log, completed traces ::  87%|########6 | 130380/150370 [00:21<00:02, 6990.26it/s]","\rparsing log, completed traces ::  87%|########7 | 131104/150370 [00:21<00:02, 7048.33it/s]","\rparsing log, completed traces ::  88%|########7 | 131899/150370 [00:21<00:02, 7312.00it/s]","\rparsing log, completed traces ::  88%|########8 | 132737/150370 [00:21<00:02, 7619.15it/s]","\rparsing log, completed traces ::  89%|########8 | 133534/150370 [00:21<00:02, 7716.13it/s]","\rparsing log, completed traces ::  89%|########9 | 134313/150370 [00:21<00:02, 7613.70it/s]","\rparsing log, completed traces ::  90%|########9 | 135080/150370 [00:21<00:02, 7553.10it/s]","\rparsing log, completed traces ::  90%|######### | 135839/150370 [00:21<00:01, 7441.33it/s]","\rparsing log, completed traces ::  91%|######### | 136586/150370 [00:21<00:01, 7389.64it/s]","\rparsing log, completed traces ::  91%|#########1| 137327/150370 [00:21<00:01, 7322.05it/s]","\rparsing log, completed traces ::  92%|#########1| 138061/150370 [00:22<00:01, 7288.60it/s]","\rparsing log, completed traces ::  92%|#########2| 138791/150370 [00:22<00:01, 7288.09it/s]","\rparsing log, completed traces ::  93%|#########2| 139521/150370 [00:22<00:01, 7266.47it/s]","\rparsing log, completed traces ::  93%|#########3| 140261/150370 [00:22<00:01, 7304.50it/s]","\rparsing log, completed traces ::  94%|#########3| 141023/150370 [00:22<00:01, 7385.70it/s]","\rparsing log, completed traces ::  94%|#########4| 141785/150370 [00:22<00:01, 7440.27it/s]","\rparsing log, completed traces ::  95%|#########4| 142538/150370 [00:22<00:01, 7465.55it/s]","\rparsing log, completed traces ::  95%|#########5| 143285/150370 [00:22<00:00, 7456.33it/s]","\rparsing log, completed traces ::  96%|#########5| 144042/150370 [00:22<00:00, 7466.92it/s]","\rparsing log, completed traces ::  96%|#########6| 144789/150370 [00:22<00:00, 7417.77it/s]","\rparsing log, completed traces ::  97%|#########6| 145531/150370 [00:23<00:00, 7023.59it/s]","\rparsing log, completed traces ::  97%|#########7| 146238/150370 [00:23<00:00, 6638.81it/s]","\rparsing log, completed traces ::  98%|#########7| 146909/150370 [00:23<00:00, 6445.51it/s]","\rparsing log, completed traces ::  98%|#########8| 147558/150370 [00:23<00:00, 6300.32it/s]","\rparsing log, completed traces ::  99%|#########8| 148191/150370 [00:23<00:00, 6191.55it/s]","\rparsing log, completed traces ::  99%|#########8| 148830/150370 [00:23<00:00, 6236.26it/s]","\rparsing log, completed traces ::  99%|#########9| 149456/150370 [00:24<00:00, 1854.09it/s]","\rparsing log, completed traces :: 100%|#########9| 150163/150370 [00:24<00:00, 2419.19it/s]","","\rparsing log, completed traces :: 100%|##########| 150370/150370 [00:24<00:00, 6092.97it/s]","\n","[data] Loaded datasets: ['BPI2012', 'BPI2017', 'ROAD']","\n","\n=== Dataset: BPI2012 ===","\n","Samples train/val/test: 22252/4176/4196; vocab=23","\n","Epoch 1: validation_loss = 0.5594 | val_acc=0.7409 | val_f1=0.5316 | val_top3=0.9840","\n","Epoch 2: validation_loss = 0.5308 | val_acc=0.7598 | val_f1=0.5425 | val_top3=0.9854","\n","Epoch 3: validation_loss = 0.5333 | val_acc=0.7574 | val_f1=0.5725 | val_top3=0.9859","\n","Epoch 4: validation_loss = 0.5195 | val_acc=0.7629 | val_f1=0.5835 | val_top3=0.9861","\n","Epoch 5: validation_loss = 0.5221 | val_acc=0.7450 | val_f1=0.6007 | val_top3=0.9854","\n","Epoch 6: validation_loss = 0.5249 | val_acc=0.7634 | val_f1=0.5833 | val_top3=0.9861","\n","Epoch 7: validation_loss = 0.5073 | val_acc=0.7639 | val_f1=0.5909 | val_top3=0.9856","\n","Epoch 8: validation_loss = 0.5113 | val_acc=0.7593 | val_f1=0.5790 | val_top3=0.9859","\n","Epoch 9: validation_loss = 0.5105 | val_acc=0.7639 | val_f1=0.5842 | val_top3=0.9852","\n","Epoch 10: validation_loss = 0.5201 | val_acc=0.7634 | val_f1=0.5797 | val_top3=0.9847","\n","[BPI2012] Train: loss=0.5148 acc=0.7777 f1=0.5609 top3=0.9868","\n","[BPI2012] Test:  loss=0.5355 acc=0.7569 f1=0.5872 top3=0.9874","\n","\n=== Dataset: BPI2017 ===","\n","Samples train/val/test: 34519/7403/7374; vocab=24","\n","Epoch 1: validation_loss = 0.4172 | val_acc=0.8368 | val_f1=0.5299 | val_top3=0.9904","\n","Epoch 2: validation_loss = 0.4004 | val_acc=0.8367 | val_f1=0.5595 | val_top3=0.9907","\n","Epoch 3: validation_loss = 0.3864 | val_acc=0.8384 | val_f1=0.5719 | val_top3=0.9912","\n","Epoch 4: validation_loss = 0.3847 | val_acc=0.8395 | val_f1=0.5903 | val_top3=0.9919","\n","Epoch 5: validation_loss = 0.3812 | val_acc=0.8405 | val_f1=0.5856 | val_top3=0.9922","\n","Epoch 6: validation_loss = 0.3802 | val_acc=0.8376 | val_f1=0.5896 | val_top3=0.9916","\n","Epoch 7: validation_loss = 0.3769 | val_acc=0.8361 | val_f1=0.5957 | val_top3=0.9924","\n","Epoch 8: validation_loss = 0.3799 | val_acc=0.8386 | val_f1=0.6180 | val_top3=0.9927","\n","Epoch 9: validation_loss = 0.3805 | val_acc=0.8391 | val_f1=0.5989 | val_top3=0.9926","\n","Epoch 10: validation_loss = 0.3756 | val_acc=0.8399 | val_f1=0.6155 | val_top3=0.9928","\n","[BPI2017] Train: loss=0.3607 acc=0.8422 f1=0.5721 top3=0.9941","\n","[BPI2017] Test:  loss=0.3877 acc=0.8332 f1=0.5710 top3=0.9906","\n","\n=== Dataset: ROAD ===","\n","Samples train/val/test: 8707/1928/1869; vocab=11","\n","Epoch 1: validation_loss = 0.4913 | val_acc=0.7993 | val_f1=0.3684 | val_top3=0.9922","\n","Epoch 2: validation_loss = 0.4561 | val_acc=0.8086 | val_f1=0.4514 | val_top3=0.9948","\n","Epoch 3: validation_loss = 0.4566 | val_acc=0.8091 | val_f1=0.5300 | val_top3=0.9969","\n","Epoch 4: validation_loss = 0.4482 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 5: validation_loss = 0.4355 | val_acc=0.8112 | val_f1=0.6427 | val_top3=0.9969","\n","Epoch 6: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 7: validation_loss = 0.4328 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 8: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 9: validation_loss = 0.4274 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 10: validation_loss = 0.4329 | val_acc=0.8122 | val_f1=0.6664 | val_top3=0.9969","\n","[ROAD] Train: loss=0.4662 acc=0.7894 f1=0.5395 top3=0.9986","\n","[ROAD] Test:  loss=0.4833 acc=0.8020 f1=0.4740 top3=0.9936","\n","Execution time: 3 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"I will load the numpy file from the specified working directory, parse the nested experiment_data structure, and compute final or best metrics as requested. For training and test, I will report the final metrics saved in the structure. For validation, I will compute the best values across epochs for accuracy, F1 score, and top-3 accuracy, and the minimum validation loss using the recorded loss history. The script prints the dataset name followed by clearly labeled metrics and executes immediately.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef safe_get(d, *keys, default=None):\n    cur = d\n    for k in keys:\n        if isinstance(cur, dict) and k in cur:\n            cur = cur[k]\n        else:\n            return default\n    return cur\n\n\ndef load_experiment_data():\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    path = os.path.join(working_dir, \"experiment_data.npy\")\n    if not os.path.isfile(path):\n        raise FileNotFoundError(f\"File not found: {path}\")\n    data = np.load(path, allow_pickle=True).item()\n    if not isinstance(data, dict):\n        raise ValueError(\"Loaded experiment_data is not a dict.\")\n    return data\n\n\ndef extract_final_train_metrics(ds_data):\n    # Expect a 'final' entry\n    train_entries = safe_get(ds_data, \"metrics\", \"train\", default=[])\n    final = None\n    for tag, vals in train_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"train loss\"] = final.get(\"loss\", None)\n        out[\"train accuracy\"] = final.get(\"acc\", None)\n        out[\"train F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"train top-3 accuracy\"] = final.get(\"top3\", None)\n    else:\n        # Fallback: try last value from losses for loss only\n        train_losses = safe_get(ds_data, \"losses\", \"train\", default=[])\n        if train_losses:\n            out[\"train loss\"] = train_losses[-1][1]\n    return out\n\n\ndef extract_best_val_metrics(ds_data):\n    # Collect per-epoch validation metrics (acc, macro_f1, top3)\n    val_entries = safe_get(ds_data, \"metrics\", \"val\", default=[])\n    best = {\n        \"validation accuracy\": None,\n        \"validation F1 score\": None,\n        \"validation top-3 accuracy\": None,\n    }\n    for tag, vals in val_entries:\n        if isinstance(vals, dict):\n            acc = vals.get(\"acc\", None)\n            f1 = vals.get(\"macro_f1\", None)\n            top3 = vals.get(\"top3\", None)\n            if acc is not None:\n                best[\"validation accuracy\"] = (\n                    acc\n                    if best[\"validation accuracy\"] is None\n                    else max(best[\"validation accuracy\"], acc)\n                )\n            if f1 is not None:\n                best[\"validation F1 score\"] = (\n                    f1\n                    if best[\"validation F1 score\"] is None\n                    else max(best[\"validation F1 score\"], f1)\n                )\n            if top3 is not None:\n                best[\"validation top-3 accuracy\"] = (\n                    top3\n                    if best[\"validation top-3 accuracy\"] is None\n                    else max(best[\"validation top-3 accuracy\"], top3)\n                )\n    # For validation loss, use recorded losses per epoch and take the minimum\n    val_losses = safe_get(ds_data, \"losses\", \"val\", default=[])\n    if val_losses:\n        min_val_loss = min((v for (_, v) in val_losses), default=None)\n    else:\n        # fallback: check 'final' val entry if present\n        min_val_loss = None\n        for tag, vals in val_entries:\n            if tag == \"final\" and isinstance(vals, dict):\n                min_val_loss = vals.get(\"loss\", None)\n                break\n    out = {\"validation loss\": min_val_loss}\n    out.update(best)\n    return out\n\n\ndef extract_final_test_metrics(ds_data):\n    test_entries = safe_get(ds_data, \"metrics\", \"test\", default=[])\n    final = None\n    for tag, vals in test_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"test loss\"] = final.get(\"loss\", None)\n        out[\"test accuracy\"] = final.get(\"acc\", None)\n        out[\"test F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"test top-3 accuracy\"] = final.get(\"top3\", None)\n    return out\n\n\ndef format_print_metrics(dataset_name, metrics_dict):\n    print(dataset_name)\n    # Keep a consistent order for readability\n    ordered_keys = [\n        \"train loss\",\n        \"train accuracy\",\n        \"train F1 score\",\n        \"train top-3 accuracy\",\n        \"validation loss\",\n        \"validation accuracy\",\n        \"validation F1 score\",\n        \"validation top-3 accuracy\",\n        \"test loss\",\n        \"test accuracy\",\n        \"test F1 score\",\n        \"test top-3 accuracy\",\n    ]\n    printed_any = False\n    for k in ordered_keys:\n        if k in metrics_dict and metrics_dict[k] is not None:\n            val = metrics_dict[k]\n            if isinstance(val, float):\n                print(f\"{k}: {val:.4f}\")\n            else:\n                print(f\"{k}: {val}\")\n            printed_any = True\n    # Print any remaining keys that were not in the ordered list\n    for k, v in metrics_dict.items():\n        if k not in ordered_keys and v is not None:\n            if isinstance(v, float):\n                print(f\"{k}: {v:.4f}\")\n            else:\n                print(f\"{k}: {v}\")\n            printed_any = True\n    if not printed_any:\n        print(\"No metrics available\")\n\n\ndef run():\n    experiment_data = load_experiment_data()\n    for ds_name, ds_data in experiment_data.items():\n        all_metrics = {}\n        all_metrics.update(extract_final_train_metrics(ds_data))\n        all_metrics.update(extract_best_val_metrics(ds_data))\n        all_metrics.update(extract_final_test_metrics(ds_data))\n        format_print_metrics(ds_name, all_metrics)\n\n\n# Execute immediately\nrun()\n","parse_term_out":["BPI2012","\n","train loss: 0.5148","\n","train accuracy: 0.7777","\n","train F1 score: 0.5609","\n","train top-3 accuracy: 0.9868","\n","validation loss: 0.5073","\n","validation accuracy: 0.7639","\n","validation F1 score: 0.6007","\n","validation top-3 accuracy: 0.9861","\n","test loss: 0.5355","\n","test accuracy: 0.7569","\n","test F1 score: 0.5872","\n","test top-3 accuracy: 0.9874","\n","BPI2017","\n","train loss: 0.3607","\n","train accuracy: 0.8422","\n","train F1 score: 0.5721","\n","train top-3 accuracy: 0.9941","\n","validation loss: 0.3756","\n","validation accuracy: 0.8405","\n","validation F1 score: 0.6180","\n","validation top-3 accuracy: 0.9928","\n","test loss: 0.3877","\n","test accuracy: 0.8332","\n","test F1 score: 0.5710","\n","test top-3 accuracy: 0.9906","\n","ROAD","\n","train loss: 0.4662","\n","train accuracy: 0.7894","\n","train F1 score: 0.5395","\n","train top-3 accuracy: 0.9986","\n","validation loss: 0.4274","\n","validation accuracy: 0.8122","\n","validation F1 score: 0.6664","\n","validation top-3 accuracy: 0.9969","\n","test loss: 0.4833","\n","test accuracy: 0.8020","\n","test F1 score: 0.4740","\n","test top-3 accuracy: 0.9936","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":239.7621512413025,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"","exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087","metric":{"value":{"metric_names":[{"metric_name":"train loss","lower_is_better":true,"description":"Average loss on the training split.","data":[{"dataset_name":"BPI2012","final_value":0.5148,"best_value":0.5148},{"dataset_name":"BPI2017","final_value":0.3607,"best_value":0.3607},{"dataset_name":"ROAD","final_value":0.4662,"best_value":0.4662}]},{"metric_name":"train accuracy","lower_is_better":false,"description":"Classification accuracy on the training split.","data":[{"dataset_name":"BPI2012","final_value":0.7777,"best_value":0.7777},{"dataset_name":"BPI2017","final_value":0.8422,"best_value":0.8422},{"dataset_name":"ROAD","final_value":0.7894,"best_value":0.7894}]},{"metric_name":"train F1 score","lower_is_better":false,"description":"F1 score on the training split.","data":[{"dataset_name":"BPI2012","final_value":0.5609,"best_value":0.5609},{"dataset_name":"BPI2017","final_value":0.5721,"best_value":0.5721},{"dataset_name":"ROAD","final_value":0.5395,"best_value":0.5395}]},{"metric_name":"train top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the training split.","data":[{"dataset_name":"BPI2012","final_value":0.9868,"best_value":0.9868},{"dataset_name":"BPI2017","final_value":0.9941,"best_value":0.9941},{"dataset_name":"ROAD","final_value":0.9986,"best_value":0.9986}]},{"metric_name":"validation loss","lower_is_better":true,"description":"Average loss on the validation split.","data":[{"dataset_name":"BPI2012","final_value":0.5073,"best_value":0.5073},{"dataset_name":"BPI2017","final_value":0.3756,"best_value":0.3756},{"dataset_name":"ROAD","final_value":0.4274,"best_value":0.4274}]},{"metric_name":"validation accuracy","lower_is_better":false,"description":"Classification accuracy on the validation split.","data":[{"dataset_name":"BPI2012","final_value":0.7639,"best_value":0.7639},{"dataset_name":"BPI2017","final_value":0.8405,"best_value":0.8405},{"dataset_name":"ROAD","final_value":0.8122,"best_value":0.8122}]},{"metric_name":"validation F1 score","lower_is_better":false,"description":"F1 score on the validation split.","data":[{"dataset_name":"BPI2012","final_value":0.6007,"best_value":0.6007},{"dataset_name":"BPI2017","final_value":0.618,"best_value":0.618},{"dataset_name":"ROAD","final_value":0.6664,"best_value":0.6664}]},{"metric_name":"validation top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the validation split.","data":[{"dataset_name":"BPI2012","final_value":0.9861,"best_value":0.9861},{"dataset_name":"BPI2017","final_value":0.9928,"best_value":0.9928},{"dataset_name":"ROAD","final_value":0.9969,"best_value":0.9969}]},{"metric_name":"test loss","lower_is_better":true,"description":"Average loss on the test split.","data":[{"dataset_name":"BPI2012","final_value":0.5355,"best_value":0.5355},{"dataset_name":"BPI2017","final_value":0.3877,"best_value":0.3877},{"dataset_name":"ROAD","final_value":0.4833,"best_value":0.4833}]},{"metric_name":"test accuracy","lower_is_better":false,"description":"Classification accuracy on the test split.","data":[{"dataset_name":"BPI2012","final_value":0.7569,"best_value":0.7569},{"dataset_name":"BPI2017","final_value":0.8332,"best_value":0.8332},{"dataset_name":"ROAD","final_value":0.802,"best_value":0.802}]},{"metric_name":"test F1 score","lower_is_better":false,"description":"F1 score on the test split.","data":[{"dataset_name":"BPI2012","final_value":0.5872,"best_value":0.5872},{"dataset_name":"BPI2017","final_value":0.571,"best_value":0.571},{"dataset_name":"ROAD","final_value":0.474,"best_value":0.474}]},{"metric_name":"test top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the test split.","data":[{"dataset_name":"BPI2012","final_value":0.9874,"best_value":0.9874},{"dataset_name":"BPI2017","final_value":0.9906,"best_value":0.9906},{"dataset_name":"ROAD","final_value":0.9936,"best_value":0.9936}]}]},"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":false,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":["../../logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_BPI2017.png","../../logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_BPI2012.png","../../logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_ROAD.png","../../logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_BPI2017.png","../../logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_ROAD.png","../../logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_BPI2012.png"],"plot_paths":["experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_BPI2012.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_ROAD.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_ROAD.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_BPI2012.png"],"plot_analyses":[{"analysis":"The validation Top-3 accuracy on the BPI2017 dataset shows a general upward trend over the epochs, indicating that the model is improving its predictive performance over time. The fluctuations suggest that while the model is learning effectively, there may be room for optimization in terms of stability or learning rate adjustments.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_BPI2017.png"},{"analysis":"The validation Top-3 accuracy for BPI2012 shows a peak early in the training process, followed by slight oscillations, and then a decline towards the end. This pattern may suggest overfitting or that further fine-tuning is required to maintain the model's performance across all epochs.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_BPI2012.png"},{"analysis":"The loss curves for the ROAD dataset indicate a significant decrease in both training and validation loss early on, which stabilizes as the epochs progress. The decreasing trend in loss is a positive sign of effective learning, and the convergence of training and validation losses suggests good generalization.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_ROAD.png"},{"analysis":"The loss curves for BPI2017 demonstrate a sharp decline in both training and validation loss in the initial epochs, followed by stabilization. This indicates that the model is effectively learning and the convergence of the two curves indicates a balance between fitting the training data and generalizing to new data.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_BPI2017.png"},{"analysis":"The validation Top-3 accuracy for the ROAD dataset displays a strong performance, rapidly achieving near-perfect accuracy and maintaining it throughout the epochs. This suggests that the model is extremely effective for this particular dataset, potentially benefiting from the inherent characteristics of the data or the model architecture used.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/val_top3_ROAD.png"},{"analysis":"The loss curves for BPI2012 show a significant drop in both training and validation loss initially, with a steadying as epochs progress. The close alignment between training and validation loss suggests that the model is not overfitting and is generalizing well to unseen data.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/loss_curves_BPI2012.png"}],"vlm_feedback_summary":"The plots demonstrate effective learning across different datasets, with promising results in terms of accuracy and loss reduction. The ROAD dataset shows exceptional model performance, which may be due to specific data characteristics or model suitability. The BPI2012 and BPI2017 datasets show good learning trends but may require further optimization to ensure consistent high performance throughout the training process.","datasets_successfully_tested":["BPI2017","BPI2012","ROAD"],"ablation_name":null,"hyperparam_name":null,"is_seed_node":true,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"# Set random seed\nimport random\nimport numpy as np\nimport torch\n\nseed = 1\nrandom.seed(seed)\nnp.random.seed(seed)\ntorch.manual_seed(seed)\nif torch.cuda.is_available():\n    torch.cuda.manual_seed(seed)\n\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nimport random\nimport math\nimport matplotlib.pyplot as plt\n\n# Device handling (required)\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data structure\nexperiment_data = {\n    \"BPI2012\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"BPI2017\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"ROAD\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n}\n\n# Data loading utilities (use provided helper)\nfrom ai_scientist.ideas.my_research_topic import load_datasets, pick_default_dataset\n\n\n# Reproducibility\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# Build prefixes\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    # Keep only 'complete' transitions if lifecycle exists\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        df = df[df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")]\n        if len(df) == 0:\n            df = df.copy()  # fallback if empty\n            df = df.sort_values([\"case_id\", \"timestamp\"])\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    # Build activity vocab\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}  # 0 for PAD\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        # Convert to numpy arrays for safe positional indexing\n        ts_ns = (\n            pd.to_datetime(g[\"timestamp\"], utc=True).astype(\"int64\").to_numpy()\n        )  # nanoseconds\n        ts = (ts_ns // 10**9).astype(np.int64)  # seconds as numpy array\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str).tolist()], dtype=np.int64\n        )\n        # simple calendar features\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        # time deltas and since start in seconds\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(\n            np.float32\n        )  # [T,5]\n        T = len(acts_ids)\n        if T < 2:\n            continue\n        # Generate prefixes of length k (min_prefix_len..min(max_prefix_len, T-1)); target = activity at position k\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            seq_acts = acts_ids[:k].tolist()\n            seq_feats = feats[:k]\n            target = int(acts_ids[k])\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": seq_acts,\n                    \"seq_feats\": seq_feats.copy(),\n                    \"target\": target,\n                    \"last_ts\": int(ts[k - 1]),\n                    \"next_ts\": int(ts[k]),\n                }\n            )\n\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n\n    # Collect feature normalization stats over all feats (initial; will be recomputed on train split)\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n    ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n    for s in samples:\n        if s[\"seq_feats\"].shape[0] > 0:\n            s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n            s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n    return samples, act2id, id2act, pad_id\n\n\n# Time-based split by case start time\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.zeros((pad_len, self.num_cont), dtype=np.float32)\n        feats_pad = np.vstack([feats_pad, feats.astype(np.float32)])\n        attn_mask = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad, dtype=torch.long),\n            \"feats\": torch.tensor(feats_pad, dtype=torch.float32),\n            \"mask\": torch.tensor(attn_mask, dtype=torch.float32),\n            \"y\": torch.tensor(s[\"target\"], dtype=torch.long),\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(\n        self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, num_layers=1, pad_idx=0\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim,\n            hidden_size=hidden,\n            batch_first=True,\n            num_layers=num_layers,\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)  # includes PAD index\n        self.pad_idx = pad_idx\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)  # [B,T,emb]\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h_last = h[-1]\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)\n        return logits\n\n\ndef collate_fn(batch):\n    keys = batch[0].keys()\n    out = {k: torch.stack([b[k] for b in batch], dim=0) for k in keys}\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys, preds_top1, preds_probs = [], [], []\n    top3_correct = 0\n    n_total = 0\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk_idx = torch.topk(probs, k=k_val, dim=1)\n            ys.extend(batch[\"y\"].detach().cpu().tolist())\n            preds_top1.extend(top1.detach().cpu().tolist())\n            preds_probs.append(probs.detach().cpu().numpy())\n            # top-3 correctness\n            for i in range(batch[\"y\"].size(0)):\n                if batch[\"y\"][i].item() in topk_idx[i].detach().cpu().tolist():\n                    top3_correct += 1\n            n_total += batch[\"y\"].size(0)\n    avg_loss = total_loss / max(1, n_total)\n    y_true = np.array(ys)\n    y_pred = np.array(preds_top1)\n    mask = y_true != pad_idx\n    y_true = y_true[mask]\n    y_pred = y_pred[mask]\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except Exception:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n_total))\n    probs_concat = (\n        np.concatenate(preds_probs, axis=0)\n        if len(preds_probs) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return avg_loss, acc, f1, top3, y_true, y_pred, probs_concat\n\n\ndef train_one_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    # Time-based split\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    # Build samples across all to get vocab; we'll re-normalize with train stats\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    # Filter per split\n    samples_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    samples_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    samples_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # Recompute normalization using train samples only\n    if len(samples_train) > 0:\n        concat_feats = [\n            s[\"seq_feats\"] for s in samples_train if s[\"seq_feats\"].shape[0] > 0\n        ]\n        if len(concat_feats) > 0:\n            all_feats = np.concatenate(concat_feats, axis=0)\n            dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n            ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n\n            def norm_samples(samples):\n                for s in samples:\n                    if s[\"seq_feats\"].shape[0] > 0:\n                        s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                        s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n            norm_samples(samples_train)\n            norm_samples(samples_val)\n            norm_samples(samples_test)\n    print(\n        f\"Samples train/val/test: {len(samples_train)}/{len(samples_val)}/{len(samples_test)}; vocab={len(act2id)}\"\n    )\n    if len(samples_train) == 0 or len(act2id) < 2:\n        print(\"Not enough data to train. Skipping.\")\n        return\n    ds_train = PrefixDataset(\n        samples_train, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_val = PrefixDataset(\n        samples_val, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_test = PrefixDataset(\n        samples_test, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    dl_train = DataLoader(\n        ds_train,\n        batch_size=batch_size,\n        shuffle=True,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_val = DataLoader(\n        ds_val,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_test = DataLoader(\n        ds_test,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n\n    # Model\n    model = LSTMBaseline(\n        vocab_size=len(act2id),\n        emb_dim=64,\n        cont_dim=5,\n        hidden=128,\n        num_layers=1,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss().to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n\n    # Training loop\n    best_val_top3 = -1.0\n    best_state = None\n    hist = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for epoch in range(1, max_epochs + 1):\n        model.train()\n        total = 0\n        running_loss = 0.0\n        for batch in dl_train:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            running_loss += loss.item() * logits.size(0)\n            total += logits.size(0)\n        train_loss = running_loss / max(1, total)\n        val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n            model, dl_val, criterion, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        hist[\"train_loss\"].append(train_loss)\n        hist[\"val_loss\"].append(val_loss)\n        hist[\"val_top3\"].append(val_top3)\n        experiment_data[name][\"losses\"][\"train\"].append((epoch, train_loss))\n        experiment_data[name][\"losses\"][\"val\"].append((epoch, val_loss))\n        experiment_data[name][\"metrics\"][\"val\"].append(\n            (epoch, {\"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3})\n        )\n        experiment_data[name][\"epochs\"].append(epoch)\n        if val_top3 > best_val_top3:\n            best_val_top3 = val_top3\n            best_state = {k: v.cpu().clone() for k, v in model.state_dict().items()}\n\n    # Load best\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n\n    # Final eval on train/val/test\n    train_loss, train_acc, train_f1, train_top3, _, _, _ = evaluate(\n        model, dl_train, criterion, device, len(act2id), pad_id\n    )\n    val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n        model, dl_val, criterion, device, len(act2id), pad_id\n    )\n    test_loss, test_acc, test_f1, test_top3, y_true_t, y_pred_t, probs_t = evaluate(\n        model, dl_test, criterion, device, len(act2id), pad_id\n    )\n    print(\n        f\"[{name}] Train: loss={train_loss:.4f} acc={train_acc:.4f} f1={train_f1:.4f} top3={train_top3:.4f}\"\n    )\n    print(\n        f\"[{name}] Test:  loss={test_loss:.4f} acc={test_acc:.4f} f1={test_f1:.4f} top3={test_top3:.4f}\"\n    )\n\n    # Save metrics\n    experiment_data[name][\"metrics\"][\"train\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": train_loss,\n                \"acc\": train_acc,\n                \"macro_f1\": train_f1,\n                \"top3\": train_top3,\n            },\n        )\n    )\n    experiment_data[name][\"metrics\"][\"val\"].append(\n        (\n            \"final\",\n            {\"loss\": val_loss, \"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3},\n        )\n    )\n    experiment_data[name][\"metrics\"][\"test\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": test_loss,\n                \"acc\": test_acc,\n                \"macro_f1\": test_f1,\n                \"top3\": test_top3,\n            },\n        )\n    )\n    experiment_data[name][\"predictions\"] = y_pred_t.tolist()\n    experiment_data[name][\"ground_truth\"] = y_true_t.tolist()\n\n    # Plots\n    try:\n        plt.figure()\n        plt.plot(hist[\"train_loss\"], label=\"train_loss\")\n        plt.plot(hist[\"val_loss\"], label=\"val_loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"loss_curves_{name}.png\"))\n        plt.close()\n\n        plt.figure()\n        plt.plot(hist[\"val_top3\"], label=\"val_top3\")\n        plt.legend()\n        plt.title(f\"Val Top-3 Acc - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Acc\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"val_top3_{name}.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n\n    # Save confusion matrix-like data (optional)\n    try:\n        from sklearn.metrics import confusion_matrix\n\n        cm = confusion_matrix(y_true_t, y_pred_t)\n        np.save(os.path.join(working_dir, f\"cm_{name}.npy\"), cm)\n    except Exception as e:\n        print(f\"[warn] Confusion matrix failed: {e}\")\n\n\ndef main():\n    datasets = load_datasets()\n    # Loop through loaded datasets; cap to 5000 earliest cases for speed\n    for key, df in datasets.items():\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep_cases = set(starts.iloc[:5000][\"case_id\"])\n                df_small = df[df[\"case_id\"].isin(keep_cases)].copy()\n            else:\n                df_small = df\n        except Exception:\n            df_small = df\n        train_one_dataset(\n            key, df_small, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    np.savez_compressed(\n        os.path.join(working_dir, \"experiment_data_compressed.npz\"),\n        data=experiment_data,\n    )\n\n\n# Execute immediately\nmain()\n","plan":"Seed node","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import (\n    f1_score,\n    accuracy_score,\n    confusion_matrix,\n    precision_recall_curve,\n    average_precision_score,\n)\nfrom sklearn.preprocessing import label_binarize\nimport random\nimport math\nfrom collections import defaultdict\n\n# Reproducibility and device\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# --------- Data loading from local XES via pm4py ----------\ndef load_xes_folder(data_dir=\"data\"):\n    datasets = {}\n    try:\n        import pm4py\n    except Exception as e:\n        print(f\"pm4py not available: {e}\")\n        return datasets\n    if not os.path.isdir(data_dir):\n        print(f\"Data directory not found: {data_dir}\")\n        return datasets\n    for fn in os.listdir(data_dir):\n        if fn.lower().endswith(\".xes\") or fn.lower().endswith(\".xes.gz\"):\n            path = os.path.join(data_dir, fn)\n            try:\n                log = pm4py.read_xes(path)\n                df = pm4py.convert_to_dataframe(log)\n                # Standardize columns\n                # pm4py dataframe typically has case:concept:name, concept:name, time:timestamp, lifecycle:transition\n                cols = df.columns\n                case_col = (\n                    \"case:concept:name\"\n                    if \"case:concept:name\" in cols\n                    else (\"case\" if \"case\" in cols else None)\n                )\n                act_col = (\n                    \"concept:name\"\n                    if \"concept:name\" in cols\n                    else (\"activity\" if \"activity\" in cols else None)\n                )\n                ts_col = (\n                    \"time:timestamp\"\n                    if \"time:timestamp\" in cols\n                    else (\"timestamp\" if \"timestamp\" in cols else None)\n                )\n                life_col = (\n                    \"lifecycle:transition\"\n                    if \"lifecycle:transition\" in cols\n                    else (\"lifecycle\" if \"lifecycle\" in cols else None)\n                )\n                if case_col is None or act_col is None or ts_col is None:\n                    print(f\"Missing required columns in {fn}, skipping.\")\n                    continue\n                out = pd.DataFrame(\n                    {\n                        \"case_id\": df[case_col].astype(str).values,\n                        \"activity\": df[act_col].astype(str).values,\n                        \"timestamp\": pd.to_datetime(df[ts_col], utc=True),\n                    }\n                )\n                if life_col is not None:\n                    out[\"lifecycle\"] = df[life_col].astype(str).values\n                name = os.path.splitext(fn)[0]\n                datasets[name] = out\n                print(\n                    f\"Loaded {name}: {len(out)} events, {out['case_id'].nunique()} cases\"\n                )\n            except Exception as e:\n                print(f\"Failed to load {fn}: {e}\")\n    return datasets\n\n\n# --------- Prefix building and split ----------\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        mask = df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")\n        if mask.any():\n            df = df[mask]\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        if len(g) < 2:\n            continue\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        ts = (g_ts.astype(\"int64\") // 10**9).to_numpy(np.int64)\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str)], dtype=np.int64\n        )\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(np.float32)\n        T = len(acts_ids)\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": acts_ids[:k].tolist(),\n                    \"seq_feats\": feats[:k].copy(),\n                    \"target\": int(acts_ids[k]),\n                    \"prefix_len\": k,\n                }\n            )\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    for s in samples:\n        pass  # initial no norm; will norm on train split\n    return samples, act2id, id2act, pad_id\n\n\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.vstack(\n            [\n                np.zeros((pad_len, self.num_cont), dtype=np.float32),\n                feats.astype(np.float32),\n            ]\n        )\n        attn = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad).long(),\n            \"feats\": torch.tensor(feats_pad).float(),\n            \"mask\": torch.tensor(attn).float(),\n            \"y\": torch.tensor(s[\"target\"]).long(),\n            \"prefix_len\": L,\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, pad_idx=0):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim, hidden_size=hidden, batch_first=True\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h = self.dropout(h[-1])\n        return self.fc(h)\n\n\ndef collate_fn(batch):\n    out = {\n        k: (\n            torch.stack([b[k] for b in batch], 0)\n            if isinstance(batch[0][k], torch.Tensor)\n            else [b[k] for b in batch]\n        )\n        for k in batch[0].keys()\n    }\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys = []\n    yhat = []\n    probs_list = []\n    n = 0\n    top3_correct = 0\n    pref_lens = []\n    top3_flags = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk = torch.topk(probs, k=k_val, dim=1)\n            y = batch[\"y\"]\n            ys.extend(y.detach().cpu().tolist())\n            yhat.extend(top1.detach().cpu().tolist())\n            probs_list.append(probs.detach().cpu().numpy())\n            for i in range(y.size(0)):\n                flag = int(y[i].item() in topk[i].detach().cpu().tolist())\n                top3_correct += flag\n                top3_flags.append(flag)\n                pref_lens.append(int(batch[\"prefix_len\"][i].item()))\n            n += y.size(0)\n    avg_loss = total_loss / max(1, n)\n    y_true = np.array(ys)\n    y_pred = np.array(yhat)\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n))\n    probs_concat = (\n        np.concatenate(probs_list, axis=0)\n        if len(probs_list) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return (\n        avg_loss,\n        acc,\n        f1,\n        top3,\n        y_true,\n        y_pred,\n        probs_concat,\n        np.array(pref_lens),\n        np.array(top3_flags),\n    )\n\n\ndef train_on_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    s_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    s_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    s_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # normalize time features on train\n    if len(s_train) > 0:\n        feats = np.concatenate(\n            [s[\"seq_feats\"] for s in s_train if len(s[\"seq_feats\"]) > 0], axis=0\n        )\n        dt_mean, dt_std = feats[:, 0].mean(), feats[:, 0].std() + 1e-6\n        ss_mean, ss_std = feats[:, 1].mean(), feats[:, 1].std() + 1e-6\n\n        def norm(samples):\n            for s in samples:\n                if s[\"seq_feats\"].shape[0] > 0:\n                    s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                    s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n        norm(s_train)\n        norm(s_val)\n        norm(s_test)\n    print(\n        f\"Samples train/val/test: {len(s_train)}/{len(s_val)}/{len(s_test)}; vocab={len(act2id)}\"\n    )\n    if len(s_train) == 0 or len(act2id) < 2:\n        print(\"Insufficient data; skipping.\")\n        return None\n    ds_tr = PrefixDataset(s_train, pad_id, max_prefix_len, 5)\n    ds_va = PrefixDataset(s_val, pad_id, max_prefix_len, 5)\n    ds_te = PrefixDataset(s_test, pad_id, max_prefix_len, 5)\n    dl_tr = DataLoader(\n        ds_tr, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n    )\n    dl_va = DataLoader(\n        ds_va, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    dl_te = DataLoader(\n        ds_te, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    model = LSTMBaseline(\n        vocab_size=len(act2id), emb_dim=64, cont_dim=5, hidden=128, pad_idx=pad_id\n    ).to(device)\n    crit = nn.CrossEntropyLoss().to(device)\n    opt = torch.optim.Adam(model.parameters(), lr=lr)\n    best_top3 = -1.0\n    best_state = None\n    history = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for ep in range(1, max_epochs + 1):\n        model.train()\n        tot = 0\n        run_loss = 0.0\n        for batch in dl_tr:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            opt.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = crit(logits, batch[\"y\"])\n            loss.backward()\n            opt.step()\n            run_loss += loss.item() * logits.size(0)\n            tot += logits.size(0)\n        tr_loss = run_loss / max(1, tot)\n        va_loss, va_acc, va_f1, va_top3, *_ = evaluate(\n            model, dl_va, crit, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {ep}: val_loss={va_loss:.4f} acc={va_acc:.4f} f1={va_f1:.4f} top3={va_top3:.4f}\"\n        )\n        history[\"train_loss\"].append(tr_loss)\n        history[\"val_loss\"].append(va_loss)\n        history[\"val_top3\"].append(va_top3)\n        if va_top3 > best_top3:\n            best_top3 = va_top3\n            best_state = {\n                k: v.detach().cpu().clone() for k, v in model.state_dict().items()\n            }\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n    tr_loss, tr_acc, tr_f1, tr_top3, *_ = evaluate(\n        model, dl_tr, crit, device, len(act2id), pad_id\n    )\n    te_loss, te_acc, te_f1, te_top3, y_true, y_pred, probs, pref_lens, top3_flags = (\n        evaluate(model, dl_te, crit, device, len(act2id), pad_id)\n    )\n    print(\n        f\"[{name}] Test: loss={te_loss:.4f} acc={te_acc:.4f} f1={te_f1:.4f} top3={te_top3:.4f}\"\n    )\n    # package experiment data\n    exp = {\n        \"metrics\": {\n            \"train\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": tr_loss,\n                        \"acc\": tr_acc,\n                        \"macro_f1\": tr_f1,\n                        \"top3\": tr_top3,\n                    },\n                )\n            ],\n            \"val\": [],\n            \"test\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": te_loss,\n                        \"acc\": te_acc,\n                        \"macro_f1\": te_f1,\n                        \"top3\": te_top3,\n                    },\n                )\n            ],\n        },\n        \"losses\": {\n            \"train\": list(enumerate(history[\"train_loss\"], start=1)),\n            \"val\": list(enumerate(history[\"val_loss\"], start=1)),\n        },\n        \"predictions\": y_pred.tolist(),\n        \"ground_truth\": y_true.tolist(),\n        \"epochs\": list(range(1, len(history[\"train_loss\"]) + 1)),\n        \"probs\": probs,\n        \"prefix_lens\": pref_lens.tolist(),\n        \"top3_flags\": top3_flags.tolist(),\n        \"act2id\": act2id,\n    }\n    # plots for this dataset\n    try:\n        plt.figure()\n        plt.plot(history[\"train_loss\"], label=\"train\")\n        plt.plot(history[\"val_loss\"], label=\"val\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating loss curves for {name}: {e}\")\n        plt.close()\n    try:\n        cm = confusion_matrix(y_true, y_pred)\n        plt.figure(figsize=(5, 4))\n        plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n        plt.colorbar()\n        plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n        plt.xlabel(\"Predicted\")\n        plt.ylabel(\"True\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_confusion_matrix.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating confusion matrix for {name}: {e}\")\n        plt.close()\n    try:\n        # Top-3 accuracy vs prefix length\n        if len(pref_lens) > 0:\n            d = defaultdict(list)\n            for L, flag in zip(pref_lens, top3_flags):\n                d[int(L)].append(int(flag))\n            xs = sorted(d.keys())\n            ys = [np.mean(d[k]) for k in xs]\n            plt.figure()\n            plt.plot(xs, ys, marker=\"o\")\n            plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n            plt.xlabel(\"Prefix Length\")\n            plt.ylabel(\"Top-3 Accuracy\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_top3_vs_prefixlen.png\"))\n            plt.close()\n    except Exception as e:\n        print(f\"Error creating Top-3 vs prefix length for {name}: {e}\")\n        plt.close()\n    try:\n        # Macro PR curve (one-vs-rest); may be coarse due to many classes\n        if probs.shape[0] > 0:\n            classes = np.unique(y_true)\n            Y = label_binarize(y_true, classes=range(probs.shape[1]))\n            # only keep columns present in classes to avoid PAD\n            present = [c for c in classes]\n            if len(present) > 1:\n                precisions = []\n                recalls = []\n                aps = []\n                for c in present:\n                    p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                    ap = average_precision_score(Y[:, c], probs[:, c])\n                    precisions.append(\n                        np.interp(np.linspace(0, 1, 101), r[::-1], p[::-1])\n                    )\n                    recalls.append(np.linspace(0, 1, 101))\n                    aps.append(ap)\n                macro_p = np.mean(np.stack(precisions, 0), 0)\n                macro_r = np.linspace(0, 1, 101)\n                plt.figure()\n                plt.plot(macro_r, macro_p, label=f\"Macro-PR (mAP={np.mean(aps):.3f})\")\n                plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Recall\")\n                plt.ylabel(\"Precision\")\n                plt.legend()\n                plt.tight_layout()\n                plt.savefig(os.path.join(working_dir, f\"{name}_macro_pr.png\"))\n                plt.close()\n    except Exception as e:\n        print(f\"Error creating PR curve for {name}: {e}\")\n        plt.close()\n    return name, exp\n\n\ndef main():\n    datasets = load_xes_folder(data_dir=os.path.join(os.getcwd(), \"data\"))\n    experiment_data = {}\n    for name, df in datasets.items():\n        # optional cap earliest 5000 cases\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep = set(starts.iloc[:5000][\"case_id\"])\n                df = df[df[\"case_id\"].isin(keep)].copy()\n        except:\n            pass\n        res = train_on_dataset(\n            name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n        if res is not None:\n            k, exp = res\n            experiment_data[k] = exp\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Print evaluation metrics\n    for k, v in experiment_data.items():\n        test_metrics = dict(v[\"metrics\"][\"test\"][0][1])\n        print(\n            f\"{k} | Test acc={test_metrics['acc']:.4f} macro_f1={test_metrics['macro_f1']:.4f} top3={test_metrics['top3']:.4f} loss={test_metrics['loss']:.4f}\"\n        )\n\n    # Secondary plotting pass strictly from experiment_data.npy (as required)\n    try:\n        experiment_data_loaded = np.load(\n            os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n        ).item()\n    except Exception as e:\n        print(f\"Error loading experiment data: {e}\")\n        experiment_data_loaded = {}\n    for name, ed in experiment_data_loaded.items():\n        try:\n            # re-plot loss curves from saved data\n            plt.figure()\n            tl = [y for (_, y) in ed.get(\"losses\", {}).get(\"train\", [])]\n            vl = [y for (_, y) in ed.get(\"losses\", {}).get(\"val\", [])]\n            if len(tl) > 0:\n                plt.plot(tl, label=\"train\")\n            if len(vl) > 0:\n                plt.plot(vl, label=\"val\")\n            plt.legend()\n            plt.title(f\"Loss Curves - {name}\\nNext-activity\")\n            plt.xlabel(\"Epoch\")\n            plt.ylabel(\"Loss\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves_reload.png\"))\n            plt.close()\n        except Exception as e:\n            print(f\"Error creating plot1: {e}\")\n            plt.close()\n        try:\n            # confusion matrix from predictions and ground truth\n            y_true = ed.get(\"ground_truth\", [])\n            y_pred = ed.get(\"predictions\", [])\n            if len(y_true) > 0 and len(y_pred) > 0:\n                cm = confusion_matrix(y_true, y_pred)\n                plt.figure(figsize=(5, 4))\n                plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n                plt.colorbar()\n                plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Predicted\")\n                plt.ylabel(\"True\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_confusion_matrix_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot2: {e}\")\n            plt.close()\n        try:\n            # Top-3 vs prefix length if present\n            pref = ed.get(\"prefix_lens\", [])\n            flags = ed.get(\"top3_flags\", [])\n            if len(pref) > 0 and len(flags) > 0:\n                d = defaultdict(list)\n                for L, f in zip(pref, flags):\n                    d[int(L)].append(int(f))\n                xs = sorted(d.keys())\n                ys = [float(np.mean(d[x])) for x in xs]\n                plt.figure()\n                plt.plot(xs, ys, marker=\"o\")\n                plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n                plt.xlabel(\"Prefix Length\")\n                plt.ylabel(\"Top-3 Accuracy\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_top3_vs_prefixlen_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot3: {e}\")\n            plt.close()\n        try:\n            # Macro PR curve if probs available\n            probs = np.array(ed.get(\"probs\", []))\n            y_true = ed.get(\"ground_truth\", [])\n            if probs.size > 0 and len(y_true) > 0:\n                classes = sorted(set(y_true))\n                Y = label_binarize(np.array(y_true), classes=range(probs.shape[1]))\n                present = classes\n                if len(present) > 1:\n                    precisions = []\n                    aps = []\n                    grid = np.linspace(0, 1, 101)\n                    for c in present:\n                        p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                        precisions.append(np.interp(grid, r[::-1], p[::-1]))\n                        aps.append(average_precision_score(Y[:, c], probs[:, c]))\n                    macro_p = np.mean(np.stack(precisions, 0), 0)\n                    plt.figure()\n                    plt.plot(grid, macro_p, label=f\"mAP={np.mean(aps):.3f}\")\n                    plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                    plt.xlabel(\"Recall\")\n                    plt.ylabel(\"Precision\")\n                    plt.legend()\n                    plt.tight_layout()\n                    plt.savefig(\n                        os.path.join(working_dir, f\"{name}_macro_pr_reload.png\")\n                    )\n                    plt.close()\n        except Exception as e:\n            print(f\"Error creating plot4: {e}\")\n            plt.close()\n\n\nif __name__ == \"__main__\":\n    main()\n","plot_plan":null,"step":7,"id":"dba5727334664144994f0a47a97edc79","ctime":1757758298.9837606,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available in /workspace/data: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<22:25,  9.72it/s]","\rparsing log, completed traces ::   1%|1         | 152/13087 [00:00<00:14, 881.45it/s]","\rparsing log, completed traces ::   2%|2         | 302/13087 [00:00<00:11, 1158.90it/s]","\rparsing log, completed traces ::   3%|3         | 455/13087 [00:00<00:09, 1303.67it/s]","\rparsing log, completed traces ::   5%|4         | 617/13087 [00:00<00:08, 1416.70it/s]","\rparsing log, completed traces ::   6%|6         | 796/13087 [00:00<00:07, 1542.06it/s]","\rparsing log, completed traces ::   7%|7         | 953/13087 [00:00<00:07, 1548.87it/s]","\rparsing log, completed traces ::   8%|8         | 1109/13087 [00:00<00:07, 1548.17it/s]","\rparsing log, completed traces ::  10%|9         | 1272/13087 [00:00<00:07, 1572.21it/s]","\rparsing log, completed traces ::  11%|#1        | 1469/13087 [00:01<00:06, 1688.00it/s]","\rparsing log, completed traces ::  13%|#2        | 1643/13087 [00:01<00:06, 1699.24it/s]","\rparsing log, completed traces ::  14%|#3        | 1813/13087 [00:01<00:06, 1693.80it/s]","\rparsing log, completed traces ::  15%|#5        | 1983/13087 [00:01<00:06, 1635.31it/s]","\rparsing log, completed traces ::  16%|#6        | 2147/13087 [00:01<00:06, 1618.88it/s]","\rparsing log, completed traces ::  18%|#7        | 2313/13087 [00:01<00:06, 1629.52it/s]","\rparsing log, completed traces ::  19%|#9        | 2492/13087 [00:01<00:12, 830.37it/s] ","\rparsing log, completed traces ::  20%|##        | 2655/13087 [00:02<00:10, 968.18it/s]","\rparsing log, completed traces ::  21%|##1       | 2797/13087 [00:02<00:09, 1052.52it/s]","\rparsing log, completed traces ::  23%|##2       | 2948/13087 [00:02<00:08, 1151.21it/s]","\rparsing log, completed traces ::  24%|##3       | 3107/13087 [00:02<00:08, 1247.32it/s]","\rparsing log, completed traces ::  25%|##5       | 3312/13087 [00:02<00:06, 1449.88it/s]","\rparsing log, completed traces ::  27%|##6       | 3522/13087 [00:02<00:05, 1618.71it/s]","\rparsing log, completed traces ::  28%|##8       | 3703/13087 [00:02<00:05, 1667.80it/s]","\rparsing log, completed traces ::  30%|##9       | 3881/13087 [00:02<00:05, 1651.24it/s]","\rparsing log, completed traces ::  31%|###1      | 4101/13087 [00:02<00:04, 1804.18it/s]","\rparsing log, completed traces ::  33%|###3      | 4338/13087 [00:02<00:04, 1963.71it/s]","\rparsing log, completed traces ::  35%|###4      | 4554/13087 [00:03<00:04, 2011.78it/s]","\rparsing log, completed traces ::  36%|###6      | 4759/13087 [00:03<00:04, 1943.14it/s]","\rparsing log, completed traces ::  38%|###7      | 4957/13087 [00:03<00:04, 1890.24it/s]","\rparsing log, completed traces ::  39%|###9      | 5149/13087 [00:03<00:04, 1832.95it/s]","\rparsing log, completed traces ::  41%|####      | 5335/13087 [00:03<00:04, 1783.17it/s]","\rparsing log, completed traces ::  42%|####2     | 5515/13087 [00:03<00:04, 1676.18it/s]","\rparsing log, completed traces ::  43%|####3     | 5685/13087 [00:03<00:04, 1679.48it/s]","\rparsing log, completed traces ::  45%|####4     | 5855/13087 [00:03<00:04, 1659.62it/s]","\rparsing log, completed traces ::  46%|####6     | 6022/13087 [00:04<00:08, 794.40it/s] ","\rparsing log, completed traces ::  47%|####7     | 6178/13087 [00:04<00:07, 917.44it/s]","\rparsing log, completed traces ::  49%|####8     | 6358/13087 [00:04<00:06, 1081.16it/s]","\rparsing log, completed traces ::  50%|####9     | 6540/13087 [00:04<00:05, 1233.35it/s]","\rparsing log, completed traces ::  51%|#####1    | 6698/13087 [00:04<00:04, 1294.12it/s]","\rparsing log, completed traces ::  52%|#####2    | 6863/13087 [00:04<00:04, 1372.10it/s]","\rparsing log, completed traces ::  54%|#####3    | 7020/13087 [00:04<00:04, 1390.27it/s]","\rparsing log, completed traces ::  55%|#####4    | 7175/13087 [00:05<00:04, 1429.68it/s]","\rparsing log, completed traces ::  56%|#####6    | 7332/13087 [00:05<00:03, 1465.80it/s]","\rparsing log, completed traces ::  57%|#####7    | 7486/13087 [00:05<00:03, 1444.11it/s]","\rparsing log, completed traces ::  58%|#####8    | 7638/13087 [00:05<00:03, 1462.94it/s]","\rparsing log, completed traces ::  60%|#####9    | 7801/13087 [00:05<00:03, 1510.14it/s]","\rparsing log, completed traces ::  61%|######    | 7955/13087 [00:05<00:03, 1494.43it/s]","\rparsing log, completed traces ::  62%|######2   | 8119/13087 [00:05<00:03, 1534.38it/s]","\rparsing log, completed traces ::  63%|######3   | 8291/13087 [00:05<00:03, 1588.44it/s]","\rparsing log, completed traces ::  65%|######4   | 8463/13087 [00:05<00:02, 1620.47it/s]","\rparsing log, completed traces ::  66%|######5   | 8630/13087 [00:06<00:02, 1629.89it/s]","\rparsing log, completed traces ::  67%|######7   | 8794/13087 [00:06<00:02, 1629.85it/s]","\rparsing log, completed traces ::  68%|######8   | 8958/13087 [00:06<00:02, 1622.75it/s]","\rparsing log, completed traces ::  70%|######9   | 9121/13087 [00:06<00:02, 1558.40it/s]","\rparsing log, completed traces ::  71%|#######1  | 9310/13087 [00:06<00:02, 1654.13it/s]","\rparsing log, completed traces ::  72%|#######2  | 9477/13087 [00:06<00:04, 730.54it/s] ","\rparsing log, completed traces ::  73%|#######3  | 9618/13087 [00:07<00:04, 834.92it/s]","\rparsing log, completed traces ::  75%|#######4  | 9778/13087 [00:07<00:03, 973.14it/s]","\rparsing log, completed traces ::  76%|#######5  | 9945/13087 [00:07<00:02, 1113.30it/s]","\rparsing log, completed traces ::  77%|#######7  | 10105/13087 [00:07<00:02, 1222.44it/s]","\rparsing log, completed traces ::  78%|#######8  | 10256/13087 [00:07<00:02, 1271.15it/s]","\rparsing log, completed traces ::  80%|#######9  | 10410/13087 [00:07<00:01, 1339.65it/s]","\rparsing log, completed traces ::  81%|########  | 10560/13087 [00:07<00:01, 1329.59it/s]","\rparsing log, completed traces ::  82%|########1 | 10725/13087 [00:07<00:01, 1415.63it/s]","\rparsing log, completed traces ::  83%|########3 | 10911/13087 [00:07<00:01, 1536.25it/s]","\rparsing log, completed traces ::  85%|########4 | 11079/13087 [00:07<00:01, 1576.80it/s]","\rparsing log, completed traces ::  86%|########5 | 11247/13087 [00:08<00:01, 1605.94it/s]","\rparsing log, completed traces ::  87%|########7 | 11440/13087 [00:08<00:00, 1694.09it/s]","\rparsing log, completed traces ::  89%|########8 | 11613/13087 [00:08<00:00, 1674.38it/s]","\rparsing log, completed traces ::  90%|######### | 11817/13087 [00:08<00:00, 1777.28it/s]","\rparsing log, completed traces ::  92%|#########1| 12001/13087 [00:08<00:00, 1789.92it/s]","\rparsing log, completed traces ::  93%|#########3| 12185/13087 [00:08<00:00, 1802.97it/s]","\rparsing log, completed traces ::  95%|#########4| 12388/13087 [00:08<00:00, 1868.73it/s]","\rparsing log, completed traces ::  96%|#########6| 12582/13087 [00:08<00:00, 1887.84it/s]","\rparsing log, completed traces ::  98%|#########7| 12772/13087 [00:08<00:00, 1876.75it/s]","\rparsing log, completed traces ::  99%|#########9| 12961/13087 [00:08<00:00, 1878.36it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:09<00:00, 1446.03it/s]","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<4:40:37,  1.87it/s]","\rparsing log, completed traces ::   0%|          | 66/31509 [00:00<03:47, 138.20it/s]","\rparsing log, completed traces ::   0%|          | 122/31509 [00:00<02:13, 235.21it/s]","\rparsing log, completed traces ::   1%|          | 185/31509 [00:00<01:34, 331.73it/s]","\rparsing log, completed traces ::   1%|          | 246/31509 [00:00<01:17, 403.29it/s]","\rparsing log, completed traces ::   1%|          | 302/31509 [00:01<02:10, 238.53it/s]","\rparsing log, completed traces ::   1%|1         | 361/31509 [00:01<01:44, 298.77it/s]","\rparsing log, completed traces ::   1%|1         | 424/31509 [00:01<01:25, 362.27it/s]","\rparsing log, completed traces ::   2%|1         | 483/31509 [00:01<01:15, 411.55it/s]","\rparsing log, completed traces ::   2%|1         | 540/31509 [00:01<01:08, 449.10it/s]","\rparsing log, completed traces ::   2%|1         | 601/31509 [00:01<01:03, 489.80it/s]","\rparsing log, completed traces ::   2%|2         | 667/31509 [00:01<00:57, 534.39it/s]","\rparsing log, completed traces ::   2%|2         | 732/31509 [00:02<00:54, 565.39it/s]","\rparsing log, completed traces ::   3%|2         | 793/31509 [00:02<00:53, 571.92it/s]","\rparsing log, completed traces ::   3%|2         | 854/31509 [00:02<00:52, 580.62it/s]","\rparsing log, completed traces ::   3%|2         | 916/31509 [00:02<00:51, 591.36it/s]","\rparsing log, completed traces ::   3%|3         | 986/31509 [00:02<00:49, 619.65it/s]","\rparsing log, completed traces ::   3%|3         | 1050/31509 [00:02<00:49, 609.79it/s]","\rparsing log, completed traces ::   4%|3         | 1112/31509 [00:02<00:49, 612.16it/s]","\rparsing log, completed traces ::   4%|3         | 1178/31509 [00:02<00:48, 625.40it/s]","\rparsing log, completed traces ::   4%|3         | 1245/31509 [00:02<00:47, 637.54it/s]","\rparsing log, completed traces ::   4%|4         | 1310/31509 [00:02<00:48, 623.75it/s]","\rparsing log, completed traces ::   4%|4         | 1381/31509 [00:03<00:46, 645.86it/s]","\rparsing log, completed traces ::   5%|4         | 1446/31509 [00:03<00:48, 616.74it/s]","\rparsing log, completed traces ::   5%|4         | 1509/31509 [00:03<01:34, 316.33it/s]","\rparsing log, completed traces ::   5%|4         | 1571/31509 [00:03<01:21, 368.49it/s]","\rparsing log, completed traces ::   5%|5         | 1634/31509 [00:03<01:11, 419.38it/s]","\rparsing log, completed traces ::   5%|5         | 1693/31509 [00:03<01:05, 456.27it/s]","\rparsing log, completed traces ::   6%|5         | 1756/31509 [00:04<00:59, 496.88it/s]","\rparsing log, completed traces ::   6%|5         | 1815/31509 [00:04<00:57, 515.38it/s]","\rparsing log, completed traces ::   6%|5         | 1873/31509 [00:04<00:56, 525.90it/s]","\rparsing log, completed traces ::   6%|6         | 1936/31509 [00:04<00:53, 552.78it/s]","\rparsing log, completed traces ::   6%|6         | 1997/31509 [00:04<00:52, 567.12it/s]","\rparsing log, completed traces ::   7%|6         | 2057/31509 [00:04<00:51, 575.05it/s]","\rparsing log, completed traces ::   7%|6         | 2122/31509 [00:04<00:49, 595.75it/s]","\rparsing log, completed traces ::   7%|6         | 2187/31509 [00:04<00:47, 611.03it/s]","\rparsing log, completed traces ::   7%|7         | 2251/31509 [00:04<00:47, 618.09it/s]","\rparsing log, completed traces ::   7%|7         | 2316/31509 [00:04<00:46, 627.23it/s]","\rparsing log, completed traces ::   8%|7         | 2380/31509 [00:05<00:46, 619.77it/s]","\rparsing log, completed traces ::   8%|7         | 2446/31509 [00:05<00:46, 627.14it/s]","\rparsing log, completed traces ::   8%|7         | 2509/31509 [00:05<00:46, 620.47it/s]","\rparsing log, completed traces ::   8%|8         | 2579/31509 [00:05<00:45, 642.87it/s]","\rparsing log, completed traces ::   8%|8         | 2648/31509 [00:05<00:43, 656.19it/s]","\rparsing log, completed traces ::   9%|8         | 2714/31509 [00:05<00:44, 642.82it/s]","\rparsing log, completed traces ::   9%|8         | 2781/31509 [00:05<00:44, 647.99it/s]","\rparsing log, completed traces ::   9%|9         | 2846/31509 [00:06<01:31, 313.08it/s]","\rparsing log, completed traces ::   9%|9         | 2915/31509 [00:06<01:16, 376.17it/s]","\rparsing log, completed traces ::   9%|9         | 2986/31509 [00:06<01:04, 440.46it/s]","\rparsing log, completed traces ::  10%|9         | 3051/31509 [00:06<00:58, 483.93it/s]","\rparsing log, completed traces ::  10%|9         | 3113/31509 [00:06<00:55, 514.20it/s]","\rparsing log, completed traces ::  10%|#         | 3178/31509 [00:06<00:51, 546.85it/s]","\rparsing log, completed traces ::  10%|#         | 3241/31509 [00:06<00:49, 566.97it/s]","\rparsing log, completed traces ::  11%|#         | 3314/31509 [00:06<00:46, 609.45it/s]","\rparsing log, completed traces ::  11%|#         | 3384/31509 [00:06<00:44, 634.37it/s]","\rparsing log, completed traces ::  11%|#         | 3457/31509 [00:07<00:42, 660.82it/s]","\rparsing log, completed traces ::  11%|#1        | 3526/31509 [00:07<00:42, 663.65it/s]","\rparsing log, completed traces ::  11%|#1        | 3594/31509 [00:07<00:42, 657.74it/s]","\rparsing log, completed traces ::  12%|#1        | 3664/31509 [00:07<00:41, 667.33it/s]","\rparsing log, completed traces ::  12%|#1        | 3732/31509 [00:07<00:41, 668.65it/s]","\rparsing log, completed traces ::  12%|#2        | 3800/31509 [00:07<00:41, 665.57it/s]","\rparsing log, completed traces ::  12%|#2        | 3867/31509 [00:07<00:42, 653.51it/s]","\rparsing log, completed traces ::  12%|#2        | 3935/31509 [00:07<00:41, 661.10it/s]","\rparsing log, completed traces ::  13%|#2        | 4002/31509 [00:07<00:41, 663.26it/s]","\rparsing log, completed traces ::  13%|#2        | 4069/31509 [00:07<00:41, 653.54it/s]","\rparsing log, completed traces ::  13%|#3        | 4135/31509 [00:08<00:41, 654.32it/s]","\rparsing log, completed traces ::  13%|#3        | 4201/31509 [00:08<00:42, 645.23it/s]","\rparsing log, completed traces ::  14%|#3        | 4266/31509 [00:08<00:42, 636.75it/s]","\rparsing log, completed traces ::  14%|#3        | 4330/31509 [00:08<00:42, 635.96it/s]","\rparsing log, completed traces ::  14%|#3        | 4396/31509 [00:08<00:42, 642.23it/s]","\rparsing log, completed traces ::  14%|#4        | 4461/31509 [00:09<01:36, 281.63it/s]","\rparsing log, completed traces ::  14%|#4        | 4529/31509 [00:09<01:18, 343.10it/s]","\rparsing log, completed traces ::  15%|#4        | 4595/31509 [00:09<01:07, 399.78it/s]","\rparsing log, completed traces ::  15%|#4        | 4665/31509 [00:09<00:58, 460.51it/s]","\rparsing log, completed traces ::  15%|#5        | 4738/31509 [00:09<00:51, 518.98it/s]","\rparsing log, completed traces ::  15%|#5        | 4803/31509 [00:09<00:48, 549.35it/s]","\rparsing log, completed traces ::  15%|#5        | 4871/31509 [00:09<00:45, 581.33it/s]","\rparsing log, completed traces ::  16%|#5        | 4937/31509 [00:09<00:44, 599.35it/s]","\rparsing log, completed traces ::  16%|#5        | 5002/31509 [00:09<00:43, 612.34it/s]","\rparsing log, completed traces ::  16%|#6        | 5074/31509 [00:09<00:41, 640.62it/s]","\rparsing log, completed traces ::  16%|#6        | 5142/31509 [00:10<00:40, 650.11it/s]","\rparsing log, completed traces ::  17%|#6        | 5209/31509 [00:10<00:40, 643.38it/s]","\rparsing log, completed traces ::  17%|#6        | 5276/31509 [00:10<00:40, 647.94it/s]","\rparsing log, completed traces ::  17%|#6        | 5342/31509 [00:10<00:40, 650.10it/s]","\rparsing log, completed traces ::  17%|#7        | 5408/31509 [00:10<00:40, 643.86it/s]","\rparsing log, completed traces ::  17%|#7        | 5477/31509 [00:10<00:39, 654.78it/s]","\rparsing log, completed traces ::  18%|#7        | 5548/31509 [00:10<00:39, 665.27it/s]","\rparsing log, completed traces ::  18%|#7        | 5619/31509 [00:10<00:38, 675.10it/s]","\rparsing log, completed traces ::  18%|#8        | 5693/31509 [00:10<00:37, 691.99it/s]","\rparsing log, completed traces ::  18%|#8        | 5766/31509 [00:10<00:36, 700.30it/s]","\rparsing log, completed traces ::  19%|#8        | 5837/31509 [00:11<00:37, 692.69it/s]","\rparsing log, completed traces ::  19%|#8        | 5911/31509 [00:11<00:36, 706.04it/s]","\rparsing log, completed traces ::  19%|#8        | 5982/31509 [00:11<00:36, 706.44it/s]","\rparsing log, completed traces ::  19%|#9        | 6053/31509 [00:11<00:37, 679.09it/s]","\rparsing log, completed traces ::  19%|#9        | 6122/31509 [00:11<00:37, 676.10it/s]","\rparsing log, completed traces ::  20%|#9        | 6190/31509 [00:11<00:37, 674.35it/s]","\rparsing log, completed traces ::  20%|#9        | 6258/31509 [00:12<01:27, 289.31it/s]","\rparsing log, completed traces ::  20%|##        | 6321/31509 [00:12<01:13, 340.98it/s]","\rparsing log, completed traces ::  20%|##        | 6392/31509 [00:12<01:01, 406.00it/s]","\rparsing log, completed traces ::  20%|##        | 6454/31509 [00:12<00:55, 448.99it/s]","\rparsing log, completed traces ::  21%|##        | 6519/31509 [00:12<00:50, 491.33it/s]","\rparsing log, completed traces ::  21%|##        | 6581/31509 [00:12<00:48, 511.65it/s]","\rparsing log, completed traces ::  21%|##1       | 6645/31509 [00:12<00:45, 543.53it/s]","\rparsing log, completed traces ::  21%|##1       | 6710/31509 [00:12<00:43, 570.70it/s]","\rparsing log, completed traces ::  22%|##1       | 6778/31509 [00:12<00:41, 599.63it/s]","\rparsing log, completed traces ::  22%|##1       | 6842/31509 [00:13<00:40, 603.95it/s]","\rparsing log, completed traces ::  22%|##1       | 6911/31509 [00:13<00:39, 628.30it/s]","\rparsing log, completed traces ::  22%|##2       | 6976/31509 [00:13<00:39, 627.41it/s]","\rparsing log, completed traces ::  22%|##2       | 7052/31509 [00:13<00:36, 665.43it/s]","\rparsing log, completed traces ::  23%|##2       | 7122/31509 [00:13<00:36, 672.44it/s]","\rparsing log, completed traces ::  23%|##2       | 7191/31509 [00:13<00:37, 656.87it/s]","\rparsing log, completed traces ::  23%|##3       | 7258/31509 [00:13<00:37, 652.25it/s]","\rparsing log, completed traces ::  23%|##3       | 7324/31509 [00:13<00:37, 643.47it/s]","\rparsing log, completed traces ::  23%|##3       | 7389/31509 [00:13<00:37, 644.06it/s]","\rparsing log, completed traces ::  24%|##3       | 7454/31509 [00:14<00:37, 637.81it/s]","\rparsing log, completed traces ::  24%|##3       | 7531/31509 [00:14<00:35, 674.32it/s]","\rparsing log, completed traces ::  24%|##4       | 7599/31509 [00:14<00:35, 674.41it/s]","\rparsing log, completed traces ::  24%|##4       | 7667/31509 [00:14<00:36, 655.43it/s]","\rparsing log, completed traces ::  25%|##4       | 7733/31509 [00:14<00:36, 650.10it/s]","\rparsing log, completed traces ::  25%|##4       | 7799/31509 [00:14<00:37, 637.40it/s]","\rparsing log, completed traces ::  25%|##4       | 7870/31509 [00:14<00:36, 655.51it/s]","\rparsing log, completed traces ::  25%|##5       | 7936/31509 [00:14<00:35, 656.13it/s]","\rparsing log, completed traces ::  25%|##5       | 8005/31509 [00:14<00:35, 665.35it/s]","\rparsing log, completed traces ::  26%|##5       | 8072/31509 [00:14<00:35, 664.28it/s]","\rparsing log, completed traces ::  26%|##5       | 8139/31509 [00:15<00:35, 661.43it/s]","\rparsing log, completed traces ::  26%|##6       | 8206/31509 [00:15<01:31, 254.48it/s]","\rparsing log, completed traces ::  26%|##6       | 8272/31509 [00:15<01:14, 310.67it/s]","\rparsing log, completed traces ::  26%|##6       | 8342/31509 [00:15<01:01, 375.01it/s]","\rparsing log, completed traces ::  27%|##6       | 8401/31509 [00:15<00:55, 415.94it/s]","\rparsing log, completed traces ::  27%|##6       | 8469/31509 [00:16<00:48, 472.10it/s]","\rparsing log, completed traces ::  27%|##7       | 8533/31509 [00:16<00:44, 511.45it/s]","\rparsing log, completed traces ::  27%|##7       | 8597/31509 [00:16<00:42, 541.74it/s]","\rparsing log, completed traces ::  27%|##7       | 8660/31509 [00:16<00:40, 563.34it/s]","\rparsing log, completed traces ::  28%|##7       | 8727/31509 [00:16<00:38, 589.32it/s]","\rparsing log, completed traces ::  28%|##7       | 8794/31509 [00:16<00:37, 610.20it/s]","\rparsing log, completed traces ::  28%|##8       | 8859/31509 [00:16<00:36, 619.31it/s]","\rparsing log, completed traces ::  28%|##8       | 8924/31509 [00:16<00:36, 617.01it/s]","\rparsing log, completed traces ::  29%|##8       | 8989/31509 [00:16<00:36, 624.91it/s]","\rparsing log, completed traces ::  29%|##8       | 9053/31509 [00:17<00:36, 607.63it/s]","\rparsing log, completed traces ::  29%|##8       | 9118/31509 [00:17<00:36, 618.30it/s]","\rparsing log, completed traces ::  29%|##9       | 9182/31509 [00:17<00:35, 623.72it/s]","\rparsing log, completed traces ::  29%|##9       | 9245/31509 [00:17<00:36, 612.85it/s]","\rparsing log, completed traces ::  30%|##9       | 9309/31509 [00:17<00:36, 615.94it/s]","\rparsing log, completed traces ::  30%|##9       | 9377/31509 [00:17<00:34, 633.28it/s]","\rparsing log, completed traces ::  30%|##9       | 9441/31509 [00:17<00:35, 624.65it/s]","\rparsing log, completed traces ::  30%|###       | 9505/31509 [00:17<00:35, 627.65it/s]","\rparsing log, completed traces ::  30%|###       | 9569/31509 [00:17<00:34, 629.79it/s]","\rparsing log, completed traces ::  31%|###       | 9634/31509 [00:17<00:34, 634.60it/s]","\rparsing log, completed traces ::  31%|###       | 9698/31509 [00:18<00:35, 616.94it/s]","\rparsing log, completed traces ::  31%|###       | 9761/31509 [00:18<00:35, 619.78it/s]","\rparsing log, completed traces ::  31%|###1      | 9824/31509 [00:18<00:35, 617.84it/s]","\rparsing log, completed traces ::  31%|###1      | 9886/31509 [00:18<00:35, 611.89it/s]","\rparsing log, completed traces ::  32%|###1      | 9949/31509 [00:18<00:35, 615.63it/s]","\rparsing log, completed traces ::  32%|###1      | 10011/31509 [00:18<00:35, 610.36it/s]","\rparsing log, completed traces ::  32%|###1      | 10073/31509 [00:18<00:35, 599.07it/s]","\rparsing log, completed traces ::  32%|###2      | 10133/31509 [00:18<00:36, 592.18it/s]","\rparsing log, completed traces ::  32%|###2      | 10198/31509 [00:18<00:35, 606.54it/s]","\rparsing log, completed traces ::  33%|###2      | 10265/31509 [00:18<00:34, 623.86it/s]","\rparsing log, completed traces ::  33%|###2      | 10328/31509 [00:19<01:36, 218.51it/s]","\rparsing log, completed traces ::  33%|###2      | 10387/31509 [00:19<01:19, 265.75it/s]","\rparsing log, completed traces ::  33%|###3      | 10447/31509 [00:19<01:06, 316.54it/s]","\rparsing log, completed traces ::  33%|###3      | 10502/31509 [00:19<00:58, 358.36it/s]","\rparsing log, completed traces ::  34%|###3      | 10557/31509 [00:20<00:52, 397.20it/s]","\rparsing log, completed traces ::  34%|###3      | 10611/31509 [00:20<00:48, 426.75it/s]","\rparsing log, completed traces ::  34%|###3      | 10666/31509 [00:20<00:45, 455.94it/s]","\rparsing log, completed traces ::  34%|###4      | 10720/31509 [00:20<00:43, 475.90it/s]","\rparsing log, completed traces ::  34%|###4      | 10774/31509 [00:20<00:42, 492.98it/s]","\rparsing log, completed traces ::  34%|###4      | 10828/31509 [00:20<00:40, 504.75it/s]","\rparsing log, completed traces ::  35%|###4      | 10886/31509 [00:20<00:39, 525.41it/s]","\rparsing log, completed traces ::  35%|###4      | 10943/31509 [00:20<00:38, 537.95it/s]","\rparsing log, completed traces ::  35%|###4      | 11006/31509 [00:20<00:36, 562.46it/s]","\rparsing log, completed traces ::  35%|###5      | 11070/31509 [00:21<00:34, 584.18it/s]","\rparsing log, completed traces ::  35%|###5      | 11134/31509 [00:21<00:33, 600.59it/s]","\rparsing log, completed traces ::  36%|###5      | 11196/31509 [00:21<00:33, 605.57it/s]","\rparsing log, completed traces ::  36%|###5      | 11263/31509 [00:21<00:32, 624.02it/s]","\rparsing log, completed traces ::  36%|###5      | 11330/31509 [00:21<00:31, 633.25it/s]","\rparsing log, completed traces ::  36%|###6      | 11394/31509 [00:21<00:32, 617.67it/s]","\rparsing log, completed traces ::  36%|###6      | 11457/31509 [00:21<00:32, 609.55it/s]","\rparsing log, completed traces ::  37%|###6      | 11519/31509 [00:21<00:32, 606.60it/s]","\rparsing log, completed traces ::  37%|###6      | 11580/31509 [00:21<00:32, 605.24it/s]","\rparsing log, completed traces ::  37%|###6      | 11644/31509 [00:21<00:32, 613.34it/s]","\rparsing log, completed traces ::  37%|###7      | 11706/31509 [00:22<00:32, 612.07it/s]","\rparsing log, completed traces ::  37%|###7      | 11770/31509 [00:22<00:31, 618.51it/s]","\rparsing log, completed traces ::  38%|###7      | 11836/31509 [00:22<00:31, 627.73it/s]","\rparsing log, completed traces ::  38%|###7      | 11905/31509 [00:22<00:30, 644.48it/s]","\rparsing log, completed traces ::  38%|###7      | 11971/31509 [00:22<00:30, 647.97it/s]","\rparsing log, completed traces ::  38%|###8      | 12036/31509 [00:22<00:30, 641.16it/s]","\rparsing log, completed traces ::  38%|###8      | 12101/31509 [00:22<00:30, 638.51it/s]","\rparsing log, completed traces ::  39%|###8      | 12165/31509 [00:22<00:30, 631.37it/s]","\rparsing log, completed traces ::  39%|###8      | 12233/31509 [00:22<00:29, 644.16it/s]","\rparsing log, completed traces ::  39%|###9      | 12298/31509 [00:22<00:30, 636.21it/s]","\rparsing log, completed traces ::  39%|###9      | 12365/31509 [00:23<00:29, 645.24it/s]","\rparsing log, completed traces ::  39%|###9      | 12430/31509 [00:23<00:29, 644.11it/s]","\rparsing log, completed traces ::  40%|###9      | 12495/31509 [00:23<00:30, 630.64it/s]","\rparsing log, completed traces ::  40%|###9      | 12559/31509 [00:23<00:30, 619.92it/s]","\rparsing log, completed traces ::  40%|####      | 12622/31509 [00:24<01:27, 216.43it/s]","\rparsing log, completed traces ::  40%|####      | 12686/31509 [00:24<01:09, 269.44it/s]","\rparsing log, completed traces ::  40%|####      | 12754/31509 [00:24<00:56, 331.63it/s]","\rparsing log, completed traces ::  41%|####      | 12816/31509 [00:24<00:49, 381.24it/s]","\rparsing log, completed traces ::  41%|####      | 12874/31509 [00:24<00:44, 417.61it/s]","\rparsing log, completed traces ::  41%|####1     | 12933/31509 [00:24<00:40, 454.03it/s]","\rparsing log, completed traces ::  41%|####1     | 13001/31509 [00:24<00:36, 507.41it/s]","\rparsing log, completed traces ::  41%|####1     | 13067/31509 [00:24<00:33, 543.99it/s]","\rparsing log, completed traces ::  42%|####1     | 13133/31509 [00:24<00:32, 573.40it/s]","\rparsing log, completed traces ::  42%|####1     | 13196/31509 [00:25<00:31, 575.95it/s]","\rparsing log, completed traces ::  42%|####2     | 13258/31509 [00:25<00:31, 579.89it/s]","\rparsing log, completed traces ::  42%|####2     | 13321/31509 [00:25<00:30, 593.91it/s]","\rparsing log, completed traces ::  42%|####2     | 13384/31509 [00:25<00:30, 602.51it/s]","\rparsing log, completed traces ::  43%|####2     | 13446/31509 [00:25<00:30, 598.00it/s]","\rparsing log, completed traces ::  43%|####2     | 13509/31509 [00:25<00:29, 605.15it/s]","\rparsing log, completed traces ::  43%|####3     | 13571/31509 [00:25<00:29, 605.38it/s]","\rparsing log, completed traces ::  43%|####3     | 13633/31509 [00:25<00:30, 593.98it/s]","\rparsing log, completed traces ::  43%|####3     | 13693/31509 [00:25<00:29, 594.72it/s]","\rparsing log, completed traces ::  44%|####3     | 13760/31509 [00:25<00:28, 613.44it/s]","\rparsing log, completed traces ::  44%|####3     | 13824/31509 [00:26<00:28, 619.23it/s]","\rparsing log, completed traces ::  44%|####4     | 13887/31509 [00:26<00:28, 615.32it/s]","\rparsing log, completed traces ::  44%|####4     | 13954/31509 [00:26<00:27, 628.69it/s]","\rparsing log, completed traces ::  44%|####4     | 14017/31509 [00:26<00:27, 628.68it/s]","\rparsing log, completed traces ::  45%|####4     | 14084/31509 [00:26<00:27, 639.96it/s]","\rparsing log, completed traces ::  45%|####4     | 14150/31509 [00:26<00:26, 642.98it/s]","\rparsing log, completed traces ::  45%|####5     | 14215/31509 [00:26<00:26, 640.63it/s]","\rparsing log, completed traces ::  45%|####5     | 14281/31509 [00:26<00:26, 645.06it/s]","\rparsing log, completed traces ::  46%|####5     | 14348/31509 [00:26<00:26, 651.08it/s]","\rparsing log, completed traces ::  46%|####5     | 14417/31509 [00:26<00:25, 658.62it/s]","\rparsing log, completed traces ::  46%|####5     | 14483/31509 [00:27<00:26, 654.65it/s]","\rparsing log, completed traces ::  46%|####6     | 14549/31509 [00:27<00:26, 631.17it/s]","\rparsing log, completed traces ::  46%|####6     | 14613/31509 [00:27<00:26, 628.37it/s]","\rparsing log, completed traces ::  47%|####6     | 14676/31509 [00:27<00:26, 628.75it/s]","\rparsing log, completed traces ::  47%|####6     | 14742/31509 [00:27<00:26, 635.83it/s]","\rparsing log, completed traces ::  47%|####7     | 14812/31509 [00:27<00:25, 652.90it/s]","\rparsing log, completed traces ::  47%|####7     | 14878/31509 [00:27<00:25, 641.39it/s]","\rparsing log, completed traces ::  47%|####7     | 14951/31509 [00:27<00:24, 665.41it/s]","\rparsing log, completed traces ::  48%|####7     | 15018/31509 [00:27<00:25, 647.95it/s]","\rparsing log, completed traces ::  48%|####7     | 15091/31509 [00:28<00:24, 670.26it/s]","\rparsing log, completed traces ::  48%|####8     | 15159/31509 [00:28<00:24, 661.46it/s]","\rparsing log, completed traces ::  48%|####8     | 15226/31509 [00:28<00:24, 656.97it/s]","\rparsing log, completed traces ::  49%|####8     | 15298/31509 [00:28<00:24, 674.18it/s]","\rparsing log, completed traces ::  49%|####8     | 15366/31509 [00:29<01:13, 219.86it/s]","\rparsing log, completed traces ::  49%|####8     | 15433/31509 [00:29<00:58, 273.89it/s]","\rparsing log, completed traces ::  49%|####9     | 15501/31509 [00:29<00:48, 332.45it/s]","\rparsing log, completed traces ::  49%|####9     | 15574/31509 [00:29<00:39, 400.71it/s]","\rparsing log, completed traces ::  50%|####9     | 15642/31509 [00:29<00:34, 454.49it/s]","\rparsing log, completed traces ::  50%|####9     | 15711/31509 [00:29<00:31, 505.44it/s]","\rparsing log, completed traces ::  50%|#####     | 15777/31509 [00:29<00:29, 534.89it/s]","\rparsing log, completed traces ::  50%|#####     | 15842/31509 [00:29<00:27, 563.64it/s]","\rparsing log, completed traces ::  51%|#####     | 15917/31509 [00:29<00:25, 611.85it/s]","\rparsing log, completed traces ::  51%|#####     | 15985/31509 [00:30<00:25, 614.95it/s]","\rparsing log, completed traces ::  51%|#####     | 16055/31509 [00:30<00:24, 633.66it/s]","\rparsing log, completed traces ::  51%|#####1    | 16125/31509 [00:30<00:23, 651.26it/s]","\rparsing log, completed traces ::  51%|#####1    | 16200/31509 [00:30<00:22, 677.54it/s]","\rparsing log, completed traces ::  52%|#####1    | 16270/31509 [00:30<00:22, 675.06it/s]","\rparsing log, completed traces ::  52%|#####1    | 16339/31509 [00:30<00:22, 669.60it/s]","\rparsing log, completed traces ::  52%|#####2    | 16407/31509 [00:30<00:23, 647.54it/s]","\rparsing log, completed traces ::  52%|#####2    | 16474/31509 [00:30<00:23, 649.32it/s]","\rparsing log, completed traces ::  52%|#####2    | 16540/31509 [00:30<00:23, 638.24it/s]","\rparsing log, completed traces ::  53%|#####2    | 16607/31509 [00:30<00:23, 647.29it/s]","\rparsing log, completed traces ::  53%|#####2    | 16678/31509 [00:31<00:22, 660.16it/s]","\rparsing log, completed traces ::  53%|#####3    | 16745/31509 [00:31<00:22, 653.01it/s]","\rparsing log, completed traces ::  53%|#####3    | 16815/31509 [00:31<00:22, 665.35it/s]","\rparsing log, completed traces ::  54%|#####3    | 16883/31509 [00:31<00:21, 667.40it/s]","\rparsing log, completed traces ::  54%|#####3    | 16951/31509 [00:31<00:21, 670.68it/s]","\rparsing log, completed traces ::  54%|#####4    | 17019/31509 [00:31<00:21, 672.16it/s]","\rparsing log, completed traces ::  54%|#####4    | 17094/31509 [00:31<00:20, 694.25it/s]","\rparsing log, completed traces ::  54%|#####4    | 17164/31509 [00:31<00:20, 686.70it/s]","\rparsing log, completed traces ::  55%|#####4    | 17233/31509 [00:31<00:21, 668.80it/s]","\rparsing log, completed traces ::  55%|#####4    | 17303/31509 [00:32<00:21, 676.27it/s]","\rparsing log, completed traces ::  55%|#####5    | 17371/31509 [00:32<00:20, 674.21it/s]","\rparsing log, completed traces ::  55%|#####5    | 17446/31509 [00:32<00:20, 693.76it/s]","\rparsing log, completed traces ::  56%|#####5    | 17516/31509 [00:32<00:21, 659.02it/s]","\rparsing log, completed traces ::  56%|#####5    | 17588/31509 [00:32<00:20, 673.34it/s]","\rparsing log, completed traces ::  56%|#####6    | 17662/31509 [00:32<00:20, 689.80it/s]","\rparsing log, completed traces ::  56%|#####6    | 17732/31509 [00:32<00:20, 682.98it/s]","\rparsing log, completed traces ::  57%|#####6    | 17803/31509 [00:32<00:19, 685.88it/s]","\rparsing log, completed traces ::  57%|#####6    | 17875/31509 [00:32<00:19, 694.35it/s]","\rparsing log, completed traces ::  57%|#####6    | 17945/31509 [00:32<00:19, 689.33it/s]","\rparsing log, completed traces ::  57%|#####7    | 18017/31509 [00:33<00:19, 695.59it/s]","\rparsing log, completed traces ::  57%|#####7    | 18089/31509 [00:33<00:19, 701.51it/s]","\rparsing log, completed traces ::  58%|#####7    | 18160/31509 [00:33<00:19, 701.94it/s]","\rparsing log, completed traces ::  58%|#####7    | 18231/31509 [00:33<00:19, 688.73it/s]","\rparsing log, completed traces ::  58%|#####8    | 18300/31509 [00:33<00:19, 688.96it/s]","\rparsing log, completed traces ::  58%|#####8    | 18369/31509 [00:33<00:19, 680.05it/s]","\rparsing log, completed traces ::  59%|#####8    | 18439/31509 [00:33<00:19, 680.29it/s]","\rparsing log, completed traces ::  59%|#####8    | 18508/31509 [00:33<00:19, 680.82it/s]","\rparsing log, completed traces ::  59%|#####8    | 18577/31509 [00:34<01:06, 195.54it/s]","\rparsing log, completed traces ::  59%|#####9    | 18650/31509 [00:34<00:50, 252.75it/s]","\rparsing log, completed traces ::  59%|#####9    | 18725/31509 [00:34<00:40, 318.52it/s]","\rparsing log, completed traces ::  60%|#####9    | 18790/31509 [00:35<00:34, 370.56it/s]","\rparsing log, completed traces ::  60%|#####9    | 18858/31509 [00:35<00:29, 426.81it/s]","\rparsing log, completed traces ::  60%|######    | 18935/31509 [00:35<00:25, 498.14it/s]","\rparsing log, completed traces ::  60%|######    | 19003/31509 [00:35<00:23, 533.23it/s]","\rparsing log, completed traces ::  61%|######    | 19071/31509 [00:35<00:21, 568.78it/s]","\rparsing log, completed traces ::  61%|######    | 19141/31509 [00:35<00:20, 601.55it/s]","\rparsing log, completed traces ::  61%|######    | 19209/31509 [00:35<00:19, 616.62it/s]","\rparsing log, completed traces ::  61%|######1   | 19277/31509 [00:35<00:19, 632.36it/s]","\rparsing log, completed traces ::  61%|######1   | 19345/31509 [00:35<00:18, 644.57it/s]","\rparsing log, completed traces ::  62%|######1   | 19417/31509 [00:35<00:18, 666.10it/s]","\rparsing log, completed traces ::  62%|######1   | 19486/31509 [00:36<00:17, 672.72it/s]","\rparsing log, completed traces ::  62%|######2   | 19558/31509 [00:36<00:17, 682.38it/s]","\rparsing log, completed traces ::  62%|######2   | 19628/31509 [00:36<00:17, 678.03it/s]","\rparsing log, completed traces ::  63%|######2   | 19697/31509 [00:36<00:17, 658.14it/s]","\rparsing log, completed traces ::  63%|######2   | 19765/31509 [00:36<00:17, 663.34it/s]","\rparsing log, completed traces ::  63%|######2   | 19832/31509 [00:36<00:17, 661.50it/s]","\rparsing log, completed traces ::  63%|######3   | 19899/31509 [00:36<00:17, 649.17it/s]","\rparsing log, completed traces ::  63%|######3   | 19965/31509 [00:36<00:17, 649.56it/s]","\rparsing log, completed traces ::  64%|######3   | 20032/31509 [00:36<00:17, 654.33it/s]","\rparsing log, completed traces ::  64%|######3   | 20098/31509 [00:36<00:17, 649.50it/s]","\rparsing log, completed traces ::  64%|######3   | 20165/31509 [00:37<00:17, 654.65it/s]","\rparsing log, completed traces ::  64%|######4   | 20236/31509 [00:37<00:16, 667.82it/s]","\rparsing log, completed traces ::  64%|######4   | 20308/31509 [00:37<00:16, 680.04it/s]","\rparsing log, completed traces ::  65%|######4   | 20377/31509 [00:37<00:16, 671.44it/s]","\rparsing log, completed traces ::  65%|######4   | 20454/31509 [00:37<00:15, 695.96it/s]","\rparsing log, completed traces ::  65%|######5   | 20524/31509 [00:37<00:15, 693.64it/s]","\rparsing log, completed traces ::  65%|######5   | 20595/31509 [00:37<00:15, 697.22it/s]","\rparsing log, completed traces ::  66%|######5   | 20668/31509 [00:37<00:15, 706.81it/s]","\rparsing log, completed traces ::  66%|######5   | 20740/31509 [00:37<00:15, 709.02it/s]","\rparsing log, completed traces ::  66%|######6   | 20811/31509 [00:37<00:15, 703.12it/s]","\rparsing log, completed traces ::  66%|######6   | 20882/31509 [00:38<00:15, 692.61it/s]","\rparsing log, completed traces ::  66%|######6   | 20952/31509 [00:38<00:15, 673.32it/s]","\rparsing log, completed traces ::  67%|######6   | 21020/31509 [00:38<00:15, 665.99it/s]","\rparsing log, completed traces ::  67%|######6   | 21087/31509 [00:38<00:15, 659.19it/s]","\rparsing log, completed traces ::  67%|######7   | 21153/31509 [00:38<00:15, 650.80it/s]","\rparsing log, completed traces ::  67%|######7   | 21227/31509 [00:38<00:15, 676.34it/s]","\rparsing log, completed traces ::  68%|######7   | 21295/31509 [00:38<00:15, 674.83it/s]","\rparsing log, completed traces ::  68%|######7   | 21363/31509 [00:38<00:15, 664.78it/s]","\rparsing log, completed traces ::  68%|######8   | 21435/31509 [00:38<00:14, 679.78it/s]","\rparsing log, completed traces ::  68%|######8   | 21509/31509 [00:39<00:14, 696.20it/s]","\rparsing log, completed traces ::  68%|######8   | 21580/31509 [00:39<00:14, 699.67it/s]","\rparsing log, completed traces ::  69%|######8   | 21651/31509 [00:39<00:14, 688.10it/s]","\rparsing log, completed traces ::  69%|######8   | 21720/31509 [00:39<00:14, 686.54it/s]","\rparsing log, completed traces ::  69%|######9   | 21792/31509 [00:39<00:13, 695.51it/s]","\rparsing log, completed traces ::  69%|######9   | 21867/31509 [00:39<00:13, 708.83it/s]","\rparsing log, completed traces ::  70%|######9   | 21938/31509 [00:39<00:13, 704.08it/s]","\rparsing log, completed traces ::  70%|######9   | 22009/31509 [00:39<00:13, 697.99it/s]","\rparsing log, completed traces ::  70%|#######   | 22083/31509 [00:39<00:13, 706.09it/s]","\rparsing log, completed traces ::  70%|#######   | 22158/31509 [00:39<00:13, 718.86it/s]","\rparsing log, completed traces ::  71%|#######   | 22230/31509 [00:40<00:13, 702.73it/s]","\rparsing log, completed traces ::  71%|#######   | 22301/31509 [00:41<00:48, 188.37it/s]","\rparsing log, completed traces ::  71%|#######   | 22369/31509 [00:41<00:38, 237.78it/s]","\rparsing log, completed traces ::  71%|#######1  | 22443/31509 [00:41<00:30, 300.79it/s]","\rparsing log, completed traces ::  71%|#######1  | 22514/31509 [00:41<00:24, 361.23it/s]","\rparsing log, completed traces ::  72%|#######1  | 22582/31509 [00:41<00:21, 417.20it/s]","\rparsing log, completed traces ::  72%|#######1  | 22650/31509 [00:41<00:18, 469.86it/s]","\rparsing log, completed traces ::  72%|#######2  | 22720/31509 [00:41<00:16, 520.03it/s]","\rparsing log, completed traces ::  72%|#######2  | 22794/31509 [00:41<00:15, 570.86it/s]","\rparsing log, completed traces ::  73%|#######2  | 22865/31509 [00:41<00:14, 604.87it/s]","\rparsing log, completed traces ::  73%|#######2  | 22935/31509 [00:42<00:14, 607.30it/s]","\rparsing log, completed traces ::  73%|#######3  | 23005/31509 [00:42<00:13, 631.84it/s]","\rparsing log, completed traces ::  73%|#######3  | 23073/31509 [00:42<00:13, 639.91it/s]","\rparsing log, completed traces ::  73%|#######3  | 23141/31509 [00:42<00:13, 642.45it/s]","\rparsing log, completed traces ::  74%|#######3  | 23208/31509 [00:42<00:12, 646.37it/s]","\rparsing log, completed traces ::  74%|#######3  | 23275/31509 [00:42<00:12, 646.31it/s]","\rparsing log, completed traces ::  74%|#######4  | 23341/31509 [00:42<00:12, 646.21it/s]","\rparsing log, completed traces ::  74%|#######4  | 23407/31509 [00:42<00:12, 644.37it/s]","\rparsing log, completed traces ::  74%|#######4  | 23474/31509 [00:42<00:12, 649.23it/s]","\rparsing log, completed traces ::  75%|#######4  | 23543/31509 [00:42<00:12, 660.80it/s]","\rparsing log, completed traces ::  75%|#######4  | 23613/31509 [00:43<00:11, 670.12it/s]","\rparsing log, completed traces ::  75%|#######5  | 23681/31509 [00:43<00:11, 669.68it/s]","\rparsing log, completed traces ::  75%|#######5  | 23751/31509 [00:43<00:11, 675.17it/s]","\rparsing log, completed traces ::  76%|#######5  | 23819/31509 [00:43<00:11, 673.76it/s]","\rparsing log, completed traces ::  76%|#######5  | 23887/31509 [00:43<00:11, 674.57it/s]","\rparsing log, completed traces ::  76%|#######6  | 23955/31509 [00:43<00:11, 665.54it/s]","\rparsing log, completed traces ::  76%|#######6  | 24023/31509 [00:43<00:11, 667.03it/s]","\rparsing log, completed traces ::  76%|#######6  | 24090/31509 [00:43<00:11, 663.18it/s]","\rparsing log, completed traces ::  77%|#######6  | 24159/31509 [00:43<00:10, 669.76it/s]","\rparsing log, completed traces ::  77%|#######6  | 24227/31509 [00:43<00:10, 669.47it/s]","\rparsing log, completed traces ::  77%|#######7  | 24295/31509 [00:44<00:10, 668.96it/s]","\rparsing log, completed traces ::  77%|#######7  | 24367/31509 [00:44<00:10, 679.71it/s]","\rparsing log, completed traces ::  78%|#######7  | 24435/31509 [00:44<00:10, 660.12it/s]","\rparsing log, completed traces ::  78%|#######7  | 24507/31509 [00:44<00:10, 672.88it/s]","\rparsing log, completed traces ::  78%|#######7  | 24577/31509 [00:44<00:10, 679.61it/s]","\rparsing log, completed traces ::  78%|#######8  | 24650/31509 [00:44<00:09, 694.22it/s]","\rparsing log, completed traces ::  78%|#######8  | 24720/31509 [00:44<00:09, 687.46it/s]","\rparsing log, completed traces ::  79%|#######8  | 24789/31509 [00:44<00:09, 687.85it/s]","\rparsing log, completed traces ::  79%|#######8  | 24862/31509 [00:44<00:09, 695.38it/s]","\rparsing log, completed traces ::  79%|#######9  | 24932/31509 [00:44<00:09, 689.55it/s]","\rparsing log, completed traces ::  79%|#######9  | 25008/31509 [00:45<00:09, 707.86it/s]","\rparsing log, completed traces ::  80%|#######9  | 25079/31509 [00:45<00:09, 670.00it/s]","\rparsing log, completed traces ::  80%|#######9  | 25150/31509 [00:45<00:09, 678.84it/s]","\rparsing log, completed traces ::  80%|########  | 25223/31509 [00:45<00:09, 691.19it/s]","\rparsing log, completed traces ::  80%|########  | 25293/31509 [00:45<00:09, 664.42it/s]","\rparsing log, completed traces ::  80%|########  | 25360/31509 [00:45<00:09, 662.97it/s]","\rparsing log, completed traces ::  81%|########  | 25431/31509 [00:45<00:09, 674.10it/s]","\rparsing log, completed traces ::  81%|########  | 25500/31509 [00:45<00:08, 678.26it/s]","\rparsing log, completed traces ::  81%|########1 | 25571/31509 [00:45<00:08, 686.29it/s]","\rparsing log, completed traces ::  81%|########1 | 25640/31509 [00:46<00:08, 666.85it/s]","\rparsing log, completed traces ::  82%|########1 | 25708/31509 [00:46<00:08, 670.35it/s]","\rparsing log, completed traces ::  82%|########1 | 25776/31509 [00:46<00:08, 670.39it/s]","\rparsing log, completed traces ::  82%|########2 | 25844/31509 [00:46<00:08, 658.62it/s]","\rparsing log, completed traces ::  82%|########2 | 25911/31509 [00:46<00:08, 659.25it/s]","\rparsing log, completed traces ::  82%|########2 | 25978/31509 [00:46<00:08, 658.66it/s]","\rparsing log, completed traces ::  83%|########2 | 26044/31509 [00:46<00:08, 651.00it/s]","\rparsing log, completed traces ::  83%|########2 | 26110/31509 [00:46<00:08, 652.96it/s]","\rparsing log, completed traces ::  83%|########3 | 26178/31509 [00:46<00:08, 659.16it/s]","\rparsing log, completed traces ::  83%|########3 | 26244/31509 [00:46<00:08, 657.69it/s]","\rparsing log, completed traces ::  84%|########3 | 26314/31509 [00:47<00:07, 668.63it/s]","\rparsing log, completed traces ::  84%|########3 | 26381/31509 [00:48<00:31, 160.95it/s]","\rparsing log, completed traces ::  84%|########3 | 26447/31509 [00:48<00:24, 206.74it/s]","\rparsing log, completed traces ::  84%|########4 | 26510/31509 [00:48<00:19, 255.96it/s]","\rparsing log, completed traces ::  84%|########4 | 26573/31509 [00:48<00:15, 308.98it/s]","\rparsing log, completed traces ::  85%|########4 | 26638/31509 [00:48<00:13, 366.45it/s]","\rparsing log, completed traces ::  85%|########4 | 26703/31509 [00:48<00:11, 421.39it/s]","\rparsing log, completed traces ::  85%|########4 | 26773/31509 [00:48<00:09, 481.00it/s]","\rparsing log, completed traces ::  85%|########5 | 26839/31509 [00:48<00:08, 522.36it/s]","\rparsing log, completed traces ::  85%|########5 | 26904/31509 [00:49<00:08, 550.53it/s]","\rparsing log, completed traces ::  86%|########5 | 26968/31509 [00:49<00:07, 573.22it/s]","\rparsing log, completed traces ::  86%|########5 | 27032/31509 [00:49<00:07, 581.27it/s]","\rparsing log, completed traces ::  86%|########6 | 27100/31509 [00:49<00:07, 605.01it/s]","\rparsing log, completed traces ::  86%|########6 | 27171/31509 [00:49<00:06, 633.77it/s]","\rparsing log, completed traces ::  86%|########6 | 27240/31509 [00:49<00:06, 646.86it/s]","\rparsing log, completed traces ::  87%|########6 | 27307/31509 [00:49<00:06, 645.29it/s]","\rparsing log, completed traces ::  87%|########6 | 27373/31509 [00:49<00:06, 647.25it/s]","\rparsing log, completed traces ::  87%|########7 | 27439/31509 [00:49<00:06, 650.56it/s]","\rparsing log, completed traces ::  87%|########7 | 27505/31509 [00:49<00:06, 649.50it/s]","\rparsing log, completed traces ::  88%|########7 | 27575/31509 [00:50<00:05, 663.94it/s]","\rparsing log, completed traces ::  88%|########7 | 27643/31509 [00:50<00:05, 668.58it/s]","\rparsing log, completed traces ::  88%|########7 | 27714/31509 [00:50<00:05, 677.42it/s]","\rparsing log, completed traces ::  88%|########8 | 27785/31509 [00:50<00:05, 686.03it/s]","\rparsing log, completed traces ::  88%|########8 | 27854/31509 [00:50<00:05, 673.41it/s]","\rparsing log, completed traces ::  89%|########8 | 27922/31509 [00:50<00:05, 667.18it/s]","\rparsing log, completed traces ::  89%|########8 | 27989/31509 [00:50<00:05, 653.01it/s]","\rparsing log, completed traces ::  89%|########9 | 28057/31509 [00:50<00:05, 659.91it/s]","\rparsing log, completed traces ::  89%|########9 | 28124/31509 [00:50<00:05, 647.76it/s]","\rparsing log, completed traces ::  89%|########9 | 28189/31509 [00:50<00:05, 647.22it/s]","\rparsing log, completed traces ::  90%|########9 | 28260/31509 [00:51<00:04, 660.70it/s]","\rparsing log, completed traces ::  90%|########9 | 28327/31509 [00:51<00:04, 648.27it/s]","\rparsing log, completed traces ::  90%|######### | 28394/31509 [00:51<00:04, 652.57it/s]","\rparsing log, completed traces ::  90%|######### | 28460/31509 [00:51<00:04, 650.83it/s]","\rparsing log, completed traces ::  91%|######### | 28530/31509 [00:51<00:04, 664.10it/s]","\rparsing log, completed traces ::  91%|######### | 28600/31509 [00:51<00:04, 673.45it/s]","\rparsing log, completed traces ::  91%|######### | 28668/31509 [00:51<00:04, 666.54it/s]","\rparsing log, completed traces ::  91%|#########1| 28735/31509 [00:51<00:04, 635.15it/s]","\rparsing log, completed traces ::  91%|#########1| 28799/31509 [00:51<00:04, 624.59it/s]","\rparsing log, completed traces ::  92%|#########1| 28862/31509 [00:52<00:04, 609.92it/s]","\rparsing log, completed traces ::  92%|#########1| 28927/31509 [00:52<00:04, 619.60it/s]","\rparsing log, completed traces ::  92%|#########2| 28996/31509 [00:52<00:03, 637.75it/s]","\rparsing log, completed traces ::  92%|#########2| 29060/31509 [00:52<00:03, 628.90it/s]","\rparsing log, completed traces ::  92%|#########2| 29124/31509 [00:52<00:03, 631.39it/s]","\rparsing log, completed traces ::  93%|#########2| 29193/31509 [00:52<00:03, 646.21it/s]","\rparsing log, completed traces ::  93%|#########2| 29260/31509 [00:52<00:03, 651.55it/s]","\rparsing log, completed traces ::  93%|#########3| 29327/31509 [00:52<00:03, 653.90it/s]","\rparsing log, completed traces ::  93%|#########3| 29393/31509 [00:52<00:03, 644.12it/s]","\rparsing log, completed traces ::  93%|#########3| 29458/31509 [00:52<00:03, 625.22it/s]","\rparsing log, completed traces ::  94%|#########3| 29521/31509 [00:53<00:03, 610.19it/s]","\rparsing log, completed traces ::  94%|#########3| 29583/31509 [00:53<00:03, 608.54it/s]","\rparsing log, completed traces ::  94%|#########4| 29654/31509 [00:53<00:02, 637.63it/s]","\rparsing log, completed traces ::  94%|#########4| 29724/31509 [00:53<00:02, 650.12it/s]","\rparsing log, completed traces ::  95%|#########4| 29790/31509 [00:53<00:02, 636.43it/s]","\rparsing log, completed traces ::  95%|#########4| 29854/31509 [00:53<00:02, 635.49it/s]","\rparsing log, completed traces ::  95%|#########4| 29922/31509 [00:53<00:02, 646.40it/s]","\rparsing log, completed traces ::  95%|#########5| 29989/31509 [00:53<00:02, 651.21it/s]","\rparsing log, completed traces ::  95%|#########5| 30055/31509 [00:53<00:02, 639.74it/s]","\rparsing log, completed traces ::  96%|#########5| 30126/31509 [00:53<00:02, 654.24it/s]","\rparsing log, completed traces ::  96%|#########5| 30192/31509 [00:54<00:02, 642.94it/s]","\rparsing log, completed traces ::  96%|#########6| 30259/31509 [00:54<00:01, 649.96it/s]","\rparsing log, completed traces ::  96%|#########6| 30328/31509 [00:54<00:01, 660.56it/s]","\rparsing log, completed traces ::  96%|#########6| 30397/31509 [00:54<00:01, 668.68it/s]","\rparsing log, completed traces ::  97%|#########6| 30464/31509 [00:54<00:01, 661.80it/s]","\rparsing log, completed traces ::  97%|#########6| 30531/31509 [00:54<00:01, 660.18it/s]","\rparsing log, completed traces ::  97%|#########7| 30604/31509 [00:54<00:01, 679.71it/s]","\rparsing log, completed traces ::  97%|#########7| 30673/31509 [00:54<00:01, 681.38it/s]","\rparsing log, completed traces ::  98%|#########7| 30742/31509 [00:56<00:05, 152.87it/s]","\rparsing log, completed traces ::  98%|#########7| 30810/31509 [00:56<00:03, 198.29it/s]","\rparsing log, completed traces ::  98%|#########7| 30877/31509 [00:56<00:02, 249.91it/s]","\rparsing log, completed traces ::  98%|#########8| 30944/31509 [00:56<00:01, 306.56it/s]","\rparsing log, completed traces ::  98%|#########8| 31012/31509 [00:56<00:01, 366.61it/s]","\rparsing log, completed traces ::  99%|#########8| 31084/31509 [00:56<00:00, 432.02it/s]","\rparsing log, completed traces ::  99%|#########8| 31151/31509 [00:56<00:00, 481.35it/s]","\rparsing log, completed traces ::  99%|#########9| 31222/31509 [00:56<00:00, 532.58it/s]","\rparsing log, completed traces ::  99%|#########9| 31289/31509 [00:56<00:00, 541.75it/s]","\rparsing log, completed traces :: 100%|#########9| 31353/31509 [00:57<00:00, 562.12it/s]","\rparsing log, completed traces :: 100%|#########9| 31421/31509 [00:57<00:00, 589.83it/s]","\rparsing log, completed traces :: 100%|#########9| 31486/31509 [00:57<00:00, 599.43it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:57<00:00, 550.33it/s]","\n","[data] Loading XES: /workspace/data/Road_Traffic_Fine_Management_Process.xes","\n","\rparsing log, completed traces ::   0%|          | 0/150370 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 743/150370 [00:00<00:20, 7429.02it/s]","\rparsing log, completed traces ::   1%|1         | 1538/150370 [00:00<00:19, 7732.05it/s]","\rparsing log, completed traces ::   2%|1         | 2312/150370 [00:00<00:19, 7709.57it/s]","\rparsing log, completed traces ::   2%|2         | 3101/150370 [00:00<00:18, 7756.35it/s]","\rparsing log, completed traces ::   3%|2         | 3878/150370 [00:00<00:18, 7759.29it/s]","\rparsing log, completed traces ::   3%|3         | 4654/150370 [00:00<00:37, 3928.97it/s]","\rparsing log, completed traces ::   4%|3         | 5434/150370 [00:00<00:30, 4690.76it/s]","\rparsing log, completed traces ::   4%|4         | 6221/150370 [00:01<00:26, 5372.78it/s]","\rparsing log, completed traces ::   5%|4         | 7011/150370 [00:01<00:24, 5973.04it/s]","\rparsing log, completed traces ::   5%|5         | 7780/150370 [00:01<00:22, 6403.07it/s]","\rparsing log, completed traces ::   6%|5         | 8601/150370 [00:01<00:20, 6881.78it/s]","\rparsing log, completed traces ::   6%|6         | 9425/150370 [00:01<00:19, 7255.41it/s]","\rparsing log, completed traces ::   7%|6         | 10224/150370 [00:01<00:18, 7461.37it/s]","\rparsing log, completed traces ::   7%|7         | 11018/150370 [00:01<00:18, 7595.45it/s]","\rparsing log, completed traces ::   8%|7         | 11813/150370 [00:01<00:18, 7673.68it/s]","\rparsing log, completed traces ::   8%|8         | 12601/150370 [00:01<00:17, 7721.28it/s]","\rparsing log, completed traces ::   9%|8         | 13391/150370 [00:01<00:17, 7773.65it/s]","\rparsing log, completed traces ::   9%|9         | 14179/150370 [00:02<00:17, 7751.04it/s]","\rparsing log, completed traces ::  10%|9         | 14977/150370 [00:02<00:17, 7816.86it/s]","\rparsing log, completed traces ::  11%|#         | 15791/150370 [00:02<00:17, 7893.60it/s]","\rparsing log, completed traces ::  11%|#1        | 16584/150370 [00:02<00:34, 3930.89it/s]","\rparsing log, completed traces ::  12%|#1        | 17411/150370 [00:02<00:28, 4679.93it/s]","\rparsing log, completed traces ::  12%|#2        | 18223/150370 [00:02<00:24, 5356.99it/s]","\rparsing log, completed traces ::  13%|#2        | 19033/150370 [00:03<00:22, 5959.08it/s]","\rparsing log, completed traces ::  13%|#3        | 19836/150370 [00:03<00:20, 6456.02it/s]","\rparsing log, completed traces ::  14%|#3        | 20646/150370 [00:03<00:18, 6874.90it/s]","\rparsing log, completed traces ::  14%|#4        | 21525/150370 [00:03<00:17, 7384.95it/s]","\rparsing log, completed traces ::  15%|#4        | 22369/150370 [00:03<00:16, 7675.45it/s]","\rparsing log, completed traces ::  15%|#5        | 23212/150370 [00:03<00:16, 7881.69it/s]","\rparsing log, completed traces ::  16%|#5        | 24036/150370 [00:03<00:16, 7716.58it/s]","\rparsing log, completed traces ::  17%|#6        | 24834/150370 [00:03<00:17, 7256.38it/s]","\rparsing log, completed traces ::  17%|#7        | 25583/150370 [00:03<00:17, 7072.80it/s]","\rparsing log, completed traces ::  17%|#7        | 26307/150370 [00:04<00:18, 6835.70it/s]","\rparsing log, completed traces ::  18%|#7        | 27002/150370 [00:04<00:18, 6693.51it/s]","\rparsing log, completed traces ::  18%|#8        | 27679/150370 [00:04<00:18, 6558.40it/s]","\rparsing log, completed traces ::  19%|#8        | 28378/150370 [00:04<00:18, 6677.22it/s]","\rparsing log, completed traces ::  19%|#9        | 29157/150370 [00:04<00:17, 6993.19it/s]","\rparsing log, completed traces ::  20%|#9        | 29895/150370 [00:04<00:16, 7104.18it/s]","\rparsing log, completed traces ::  20%|##        | 30610/150370 [00:05<00:36, 3237.45it/s]","\rparsing log, completed traces ::  21%|##        | 31273/150370 [00:05<00:31, 3774.35it/s]","\rparsing log, completed traces ::  21%|##1       | 31948/150370 [00:05<00:27, 4325.99it/s]","\rparsing log, completed traces ::  22%|##1       | 32624/150370 [00:05<00:24, 4835.00it/s]","\rparsing log, completed traces ::  22%|##2       | 33301/150370 [00:05<00:22, 5276.33it/s]","\rparsing log, completed traces ::  23%|##2       | 33987/150370 [00:05<00:20, 5663.87it/s]","\rparsing log, completed traces ::  23%|##3       | 34657/150370 [00:05<00:19, 5933.92it/s]","\rparsing log, completed traces ::  24%|##3       | 35337/150370 [00:05<00:18, 6167.15it/s]","\rparsing log, completed traces ::  24%|##3       | 36011/150370 [00:05<00:18, 6326.24it/s]","\rparsing log, completed traces ::  24%|##4       | 36698/150370 [00:05<00:17, 6478.56it/s]","\rparsing log, completed traces ::  25%|##4       | 37374/150370 [00:06<00:17, 6558.93it/s]","\rparsing log, completed traces ::  25%|##5       | 38070/150370 [00:06<00:16, 6675.88it/s]","\rparsing log, completed traces ::  26%|##5       | 38787/150370 [00:06<00:16, 6813.79it/s]","\rparsing log, completed traces ::  26%|##6       | 39524/150370 [00:06<00:15, 6976.99it/s]","\rparsing log, completed traces ::  27%|##6       | 40248/150370 [00:06<00:15, 7054.64it/s]","\rparsing log, completed traces ::  27%|##7       | 40959/150370 [00:06<00:15, 7065.58it/s]","\rparsing log, completed traces ::  28%|##7       | 41718/150370 [00:06<00:15, 7221.19it/s]","\rparsing log, completed traces ::  28%|##8       | 42453/150370 [00:06<00:14, 7249.98it/s]","\rparsing log, completed traces ::  29%|##8       | 43194/150370 [00:06<00:14, 7296.44it/s]","\rparsing log, completed traces ::  29%|##9       | 43929/150370 [00:06<00:14, 7308.99it/s]","\rparsing log, completed traces ::  30%|##9       | 44661/150370 [00:07<00:14, 7309.95it/s]","\rparsing log, completed traces ::  30%|###       | 45393/150370 [00:07<00:34, 3070.08it/s]","\rparsing log, completed traces ::  31%|###       | 46104/150370 [00:07<00:28, 3685.11it/s]","\rparsing log, completed traces ::  31%|###1      | 46780/150370 [00:07<00:24, 4234.18it/s]","\rparsing log, completed traces ::  32%|###1      | 47428/150370 [00:07<00:21, 4690.09it/s]","\rparsing log, completed traces ::  32%|###1      | 48115/150370 [00:07<00:19, 5180.68it/s]","\rparsing log, completed traces ::  33%|###2      | 48919/150370 [00:08<00:17, 5869.14it/s]","\rparsing log, completed traces ::  33%|###3      | 49672/150370 [00:08<00:15, 6296.78it/s]","\rparsing log, completed traces ::  34%|###3      | 50383/150370 [00:08<00:15, 6508.25it/s]","\rparsing log, completed traces ::  34%|###3      | 51098/150370 [00:08<00:14, 6686.27it/s]","\rparsing log, completed traces ::  34%|###4      | 51823/150370 [00:08<00:14, 6843.60it/s]","\rparsing log, completed traces ::  35%|###4      | 52539/150370 [00:08<00:14, 6858.91it/s]","\rparsing log, completed traces ::  35%|###5      | 53247/150370 [00:08<00:14, 6845.69it/s]","\rparsing log, completed traces ::  36%|###5      | 53947/150370 [00:08<00:14, 6754.38it/s]","\rparsing log, completed traces ::  36%|###6      | 54659/150370 [00:08<00:13, 6858.90it/s]","\rparsing log, completed traces ::  37%|###6      | 55353/150370 [00:09<00:13, 6787.48it/s]","\rparsing log, completed traces ::  37%|###7      | 56052/150370 [00:09<00:13, 6843.88it/s]","\rparsing log, completed traces ::  38%|###7      | 56792/150370 [00:09<00:13, 7006.65it/s]","\rparsing log, completed traces ::  38%|###8      | 57545/150370 [00:09<00:12, 7156.79it/s]","\rparsing log, completed traces ::  39%|###8      | 58283/150370 [00:09<00:12, 7210.41it/s]","\rparsing log, completed traces ::  39%|###9      | 59025/150370 [00:09<00:12, 7262.61it/s]","\rparsing log, completed traces ::  40%|###9      | 59818/150370 [00:09<00:12, 7460.49it/s]","\rparsing log, completed traces ::  40%|####      | 60589/150370 [00:09<00:11, 7520.66it/s]","\rparsing log, completed traces ::  41%|####      | 61447/150370 [00:09<00:11, 7835.79it/s]","\rparsing log, completed traces ::  41%|####1     | 62286/150370 [00:09<00:11, 8000.86it/s]","\rparsing log, completed traces ::  42%|####1     | 63087/150370 [00:10<00:28, 3078.71it/s]","\rparsing log, completed traces ::  42%|####2     | 63847/150370 [00:10<00:23, 3715.73it/s]","\rparsing log, completed traces ::  43%|####2     | 64640/150370 [00:10<00:19, 4422.83it/s]","\rparsing log, completed traces ::  44%|####3     | 65423/150370 [00:10<00:16, 5072.38it/s]","\rparsing log, completed traces ::  44%|####4     | 66237/150370 [00:10<00:14, 5720.66it/s]","\rparsing log, completed traces ::  45%|####4     | 67032/150370 [00:11<00:13, 6245.16it/s]","\rparsing log, completed traces ::  45%|####5     | 67914/150370 [00:11<00:11, 6884.77it/s]","\rparsing log, completed traces ::  46%|####5     | 68723/150370 [00:11<00:11, 7201.80it/s]","\rparsing log, completed traces ::  46%|####6     | 69522/150370 [00:11<00:10, 7392.05it/s]","\rparsing log, completed traces ::  47%|####6     | 70318/150370 [00:11<00:10, 7473.34it/s]","\rparsing log, completed traces ::  47%|####7     | 71106/150370 [00:11<00:10, 7582.08it/s]","\rparsing log, completed traces ::  48%|####7     | 71893/150370 [00:11<00:10, 7561.18it/s]","\rparsing log, completed traces ::  48%|####8     | 72669/150370 [00:11<00:10, 7616.70it/s]","\rparsing log, completed traces ::  49%|####8     | 73445/150370 [00:11<00:10, 7419.68it/s]","\rparsing log, completed traces ::  49%|####9     | 74198/150370 [00:11<00:10, 7427.07it/s]","\rparsing log, completed traces ::  50%|####9     | 75073/150370 [00:12<00:09, 7788.80it/s]","\rparsing log, completed traces ::  50%|#####     | 75859/150370 [00:12<00:09, 7805.70it/s]","\rparsing log, completed traces ::  51%|#####1    | 76692/150370 [00:12<00:09, 7933.53it/s]","\rparsing log, completed traces ::  52%|#####1    | 77489/150370 [00:12<00:09, 7813.77it/s]","\rparsing log, completed traces ::  52%|#####2    | 78273/150370 [00:12<00:09, 7739.57it/s]","\rparsing log, completed traces ::  53%|#####2    | 79049/150370 [00:12<00:09, 7650.76it/s]","\rparsing log, completed traces ::  53%|#####3    | 79816/150370 [00:12<00:09, 7613.92it/s]","\rparsing log, completed traces ::  54%|#####3    | 80579/150370 [00:12<00:09, 7537.08it/s]","\rparsing log, completed traces ::  54%|#####4    | 81334/150370 [00:12<00:09, 7481.26it/s]","\rparsing log, completed traces ::  55%|#####4    | 82106/150370 [00:13<00:09, 7535.51it/s]","\rparsing log, completed traces ::  55%|#####5    | 82924/150370 [00:13<00:08, 7725.07it/s]","\rparsing log, completed traces ::  56%|#####5    | 83698/150370 [00:13<00:08, 7676.57it/s]","\rparsing log, completed traces ::  56%|#####6    | 84467/150370 [00:13<00:08, 7595.84it/s]","\rparsing log, completed traces ::  57%|#####6    | 85227/150370 [00:14<00:24, 2678.27it/s]","\rparsing log, completed traces ::  57%|#####7    | 85975/150370 [00:14<00:19, 3299.17it/s]","\rparsing log, completed traces ::  58%|#####7    | 86738/150370 [00:14<00:16, 3969.40it/s]","\rparsing log, completed traces ::  58%|#####8    | 87511/150370 [00:14<00:13, 4652.91it/s]","\rparsing log, completed traces ::  59%|#####8    | 88292/150370 [00:14<00:11, 5304.01it/s]","\rparsing log, completed traces ::  59%|#####9    | 89058/150370 [00:14<00:10, 5839.73it/s]","\rparsing log, completed traces ::  60%|#####9    | 89872/150370 [00:14<00:09, 6403.02it/s]","\rparsing log, completed traces ::  60%|######    | 90630/150370 [00:14<00:08, 6668.30it/s]","\rparsing log, completed traces ::  61%|######    | 91405/150370 [00:14<00:08, 6958.66it/s]","\rparsing log, completed traces ::  61%|######1   | 92182/150370 [00:14<00:08, 7182.17it/s]","\rparsing log, completed traces ::  62%|######1   | 92947/150370 [00:15<00:07, 7314.78it/s]","\rparsing log, completed traces ::  62%|######2   | 93722/150370 [00:15<00:07, 7439.13it/s]","\rparsing log, completed traces ::  63%|######2   | 94515/150370 [00:15<00:07, 7581.61it/s]","\rparsing log, completed traces ::  63%|######3   | 95316/150370 [00:15<00:07, 7707.23it/s]","\rparsing log, completed traces ::  64%|######3   | 96099/150370 [00:15<00:07, 7708.39it/s]","\rparsing log, completed traces ::  64%|######4   | 96879/150370 [00:15<00:06, 7696.43it/s]","\rparsing log, completed traces ::  65%|######4   | 97655/150370 [00:15<00:06, 7643.61it/s]","\rparsing log, completed traces ::  65%|######5   | 98437/150370 [00:15<00:06, 7693.39it/s]","\rparsing log, completed traces ::  66%|######5   | 99210/150370 [00:15<00:06, 7677.23it/s]","\rparsing log, completed traces ::  66%|######6   | 99980/150370 [00:15<00:06, 7618.88it/s]","\rparsing log, completed traces ::  67%|######7   | 100751/150370 [00:16<00:06, 7644.27it/s]","\rparsing log, completed traces ::  68%|######7   | 101521/150370 [00:16<00:06, 7660.00it/s]","\rparsing log, completed traces ::  68%|######8   | 102297/150370 [00:16<00:06, 7689.10it/s]","\rparsing log, completed traces ::  69%|######8   | 103067/150370 [00:16<00:06, 7679.50it/s]","\rparsing log, completed traces ::  69%|######9   | 103836/150370 [00:16<00:06, 7649.80it/s]","\rparsing log, completed traces ::  70%|######9   | 104602/150370 [00:16<00:06, 7571.97it/s]","\rparsing log, completed traces ::  70%|#######   | 105360/150370 [00:16<00:05, 7549.07it/s]","\rparsing log, completed traces ::  71%|#######   | 106116/150370 [00:16<00:05, 7496.24it/s]","\rparsing log, completed traces ::  71%|#######1  | 106866/150370 [00:16<00:05, 7479.34it/s]","\rparsing log, completed traces ::  72%|#######1  | 107615/150370 [00:16<00:05, 7455.18it/s]","\rparsing log, completed traces ::  72%|#######2  | 108361/150370 [00:17<00:05, 7393.40it/s]","\rparsing log, completed traces ::  73%|#######2  | 109101/150370 [00:17<00:05, 7391.35it/s]","\rparsing log, completed traces ::  73%|#######3  | 109841/150370 [00:17<00:17, 2381.85it/s]","\rparsing log, completed traces ::  74%|#######3  | 110581/150370 [00:18<00:13, 2983.68it/s]","\rparsing log, completed traces ::  74%|#######4  | 111320/150370 [00:18<00:10, 3623.65it/s]","\rparsing log, completed traces ::  75%|#######4  | 112077/150370 [00:18<00:08, 4297.63it/s]","\rparsing log, completed traces ::  75%|#######5  | 112796/150370 [00:18<00:07, 4865.95it/s]","\rparsing log, completed traces ::  76%|#######5  | 113562/150370 [00:18<00:06, 5480.76it/s]","\rparsing log, completed traces ::  76%|#######6  | 114447/150370 [00:18<00:05, 6282.19it/s]","\rparsing log, completed traces ::  77%|#######6  | 115360/150370 [00:18<00:04, 7006.92it/s]","\rparsing log, completed traces ::  77%|#######7  | 116253/150370 [00:18<00:04, 7517.94it/s]","\rparsing log, completed traces ::  78%|#######8  | 117304/150370 [00:18<00:03, 8342.80it/s]","\rparsing log, completed traces ::  79%|#######8  | 118203/150370 [00:19<00:04, 7896.13it/s]","\rparsing log, completed traces ::  79%|#######9  | 119042/150370 [00:19<00:04, 7617.65it/s]","\rparsing log, completed traces ::  80%|#######9  | 119840/150370 [00:19<00:04, 7403.92it/s]","\rparsing log, completed traces ::  80%|########  | 120606/150370 [00:19<00:04, 7316.87it/s]","\rparsing log, completed traces ::  81%|########  | 121355/150370 [00:19<00:04, 7210.53it/s]","\rparsing log, completed traces ::  81%|########1 | 122088/150370 [00:19<00:03, 7145.23it/s]","\rparsing log, completed traces ::  82%|########1 | 122811/150370 [00:19<00:03, 7146.64it/s]","\rparsing log, completed traces ::  82%|########2 | 123532/150370 [00:19<00:03, 7119.82it/s]","\rparsing log, completed traces ::  83%|########2 | 124248/150370 [00:19<00:03, 7044.91it/s]","\rparsing log, completed traces ::  83%|########3 | 124955/150370 [00:19<00:03, 7009.10it/s]","\rparsing log, completed traces ::  84%|########3 | 125681/150370 [00:20<00:03, 7080.32it/s]","\rparsing log, completed traces ::  84%|########4 | 126393/150370 [00:20<00:03, 7089.46it/s]","\rparsing log, completed traces ::  85%|########4 | 127123/150370 [00:20<00:03, 7151.21it/s]","\rparsing log, completed traces ::  85%|########5 | 127839/150370 [00:20<00:03, 7141.88it/s]","\rparsing log, completed traces ::  86%|########5 | 128569/150370 [00:20<00:03, 7159.13it/s]","\rparsing log, completed traces ::  86%|########5 | 129308/150370 [00:20<00:02, 7226.59it/s]","\rparsing log, completed traces ::  86%|########6 | 130031/150370 [00:20<00:02, 7227.42it/s]","\rparsing log, completed traces ::  87%|########6 | 130756/150370 [00:20<00:02, 7211.59it/s]","\rparsing log, completed traces ::  87%|########7 | 131495/150370 [00:20<00:02, 7262.77it/s]","\rparsing log, completed traces ::  88%|########8 | 132359/150370 [00:20<00:02, 7658.08it/s]","\rparsing log, completed traces ::  89%|########8 | 133166/150370 [00:21<00:02, 7771.99it/s]","\rparsing log, completed traces ::  89%|########9 | 133944/150370 [00:21<00:02, 7744.47it/s]","\rparsing log, completed traces ::  90%|########9 | 134719/150370 [00:21<00:02, 7598.07it/s]","\rparsing log, completed traces ::  90%|######### | 135480/150370 [00:21<00:02, 7436.98it/s]","\rparsing log, completed traces ::  91%|######### | 136225/150370 [00:21<00:01, 7388.21it/s]","\rparsing log, completed traces ::  91%|#########1| 136965/150370 [00:21<00:01, 7375.75it/s]","\rparsing log, completed traces ::  92%|#########1| 137703/150370 [00:21<00:01, 7281.56it/s]","\rparsing log, completed traces ::  92%|#########2| 138432/150370 [00:22<00:05, 2111.70it/s]","\rparsing log, completed traces ::  93%|#########2| 139165/150370 [00:22<00:04, 2676.78it/s]","\rparsing log, completed traces ::  93%|#########3| 139881/150370 [00:22<00:03, 3274.15it/s]","\rparsing log, completed traces ::  94%|#########3| 140663/150370 [00:22<00:02, 4003.79it/s]","\rparsing log, completed traces ::  94%|#########4| 141398/150370 [00:23<00:01, 4622.35it/s]","\rparsing log, completed traces ::  95%|#########4| 142167/150370 [00:23<00:01, 5266.63it/s]","\rparsing log, completed traces ::  95%|#########5| 142906/150370 [00:23<00:01, 5756.38it/s]","\rparsing log, completed traces ::  96%|#########5| 143655/150370 [00:23<00:01, 6185.07it/s]","\rparsing log, completed traces ::  96%|#########6| 144396/150370 [00:23<00:00, 6504.23it/s]","\rparsing log, completed traces ::  97%|#########6| 145129/150370 [00:23<00:00, 6609.69it/s]","\rparsing log, completed traces ::  97%|#########6| 145849/150370 [00:23<00:00, 6411.83it/s]","\rparsing log, completed traces ::  97%|#########7| 146532/150370 [00:23<00:00, 6240.02it/s]","\rparsing log, completed traces ::  98%|#########7| 147186/150370 [00:23<00:00, 6151.72it/s]","\rparsing log, completed traces ::  98%|#########8| 147822/150370 [00:23<00:00, 6080.40it/s]","\rparsing log, completed traces ::  99%|#########8| 148444/150370 [00:24<00:00, 6075.54it/s]","\rparsing log, completed traces ::  99%|#########9| 149091/150370 [00:24<00:00, 6185.53it/s]","\rparsing log, completed traces :: 100%|#########9| 149776/150370 [00:24<00:00, 6376.14it/s]","","\rparsing log, completed traces :: 100%|##########| 150370/150370 [00:24<00:00, 6166.56it/s]","\n","[data] Loaded datasets: ['BPI2012', 'BPI2017', 'ROAD']","\n","\n=== Dataset: BPI2012 ===","\n","Samples train/val/test: 22252/4176/4196; vocab=23","\n","Epoch 1: validation_loss = 0.5594 | val_acc=0.7409 | val_f1=0.5316 | val_top3=0.9840","\n","Epoch 2: validation_loss = 0.5308 | val_acc=0.7598 | val_f1=0.5425 | val_top3=0.9854","\n","Epoch 3: validation_loss = 0.5333 | val_acc=0.7574 | val_f1=0.5725 | val_top3=0.9859","\n","Epoch 4: validation_loss = 0.5195 | val_acc=0.7629 | val_f1=0.5835 | val_top3=0.9861","\n","Epoch 5: validation_loss = 0.5221 | val_acc=0.7450 | val_f1=0.6007 | val_top3=0.9854","\n","Epoch 6: validation_loss = 0.5249 | val_acc=0.7634 | val_f1=0.5833 | val_top3=0.9861","\n","Epoch 7: validation_loss = 0.5073 | val_acc=0.7639 | val_f1=0.5909 | val_top3=0.9856","\n","Epoch 8: validation_loss = 0.5113 | val_acc=0.7593 | val_f1=0.5790 | val_top3=0.9859","\n","Epoch 9: validation_loss = 0.5105 | val_acc=0.7639 | val_f1=0.5842 | val_top3=0.9852","\n","Epoch 10: validation_loss = 0.5201 | val_acc=0.7634 | val_f1=0.5797 | val_top3=0.9847","\n","[BPI2012] Train: loss=0.5148 acc=0.7777 f1=0.5609 top3=0.9868","\n","[BPI2012] Test:  loss=0.5355 acc=0.7569 f1=0.5872 top3=0.9874","\n","\n=== Dataset: BPI2017 ===","\n","Samples train/val/test: 34519/7403/7374; vocab=24","\n","Epoch 1: validation_loss = 0.4172 | val_acc=0.8368 | val_f1=0.5299 | val_top3=0.9904","\n","Epoch 2: validation_loss = 0.4004 | val_acc=0.8367 | val_f1=0.5595 | val_top3=0.9907","\n","Epoch 3: validation_loss = 0.3864 | val_acc=0.8384 | val_f1=0.5719 | val_top3=0.9912","\n","Epoch 4: validation_loss = 0.3847 | val_acc=0.8395 | val_f1=0.5903 | val_top3=0.9919","\n","Epoch 5: validation_loss = 0.3812 | val_acc=0.8405 | val_f1=0.5856 | val_top3=0.9922","\n","Epoch 6: validation_loss = 0.3802 | val_acc=0.8376 | val_f1=0.5896 | val_top3=0.9916","\n","Epoch 7: validation_loss = 0.3769 | val_acc=0.8361 | val_f1=0.5957 | val_top3=0.9924","\n","Epoch 8: validation_loss = 0.3799 | val_acc=0.8386 | val_f1=0.6180 | val_top3=0.9927","\n","Epoch 9: validation_loss = 0.3805 | val_acc=0.8391 | val_f1=0.5989 | val_top3=0.9926","\n","Epoch 10: validation_loss = 0.3756 | val_acc=0.8399 | val_f1=0.6155 | val_top3=0.9928","\n","[BPI2017] Train: loss=0.3607 acc=0.8422 f1=0.5721 top3=0.9941","\n","[BPI2017] Test:  loss=0.3877 acc=0.8332 f1=0.5710 top3=0.9906","\n","\n=== Dataset: ROAD ===","\n","Samples train/val/test: 8707/1928/1869; vocab=11","\n","Epoch 1: validation_loss = 0.4913 | val_acc=0.7993 | val_f1=0.3684 | val_top3=0.9922","\n","Epoch 2: validation_loss = 0.4561 | val_acc=0.8086 | val_f1=0.4514 | val_top3=0.9948","\n","Epoch 3: validation_loss = 0.4566 | val_acc=0.8091 | val_f1=0.5300 | val_top3=0.9969","\n","Epoch 4: validation_loss = 0.4482 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 5: validation_loss = 0.4355 | val_acc=0.8112 | val_f1=0.6427 | val_top3=0.9969","\n","Epoch 6: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 7: validation_loss = 0.4328 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 8: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 9: validation_loss = 0.4274 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 10: validation_loss = 0.4329 | val_acc=0.8122 | val_f1=0.6664 | val_top3=0.9969","\n","[ROAD] Train: loss=0.4662 acc=0.7894 f1=0.5395 top3=0.9986","\n","[ROAD] Test:  loss=0.4833 acc=0.8020 f1=0.4740 top3=0.9936","\n","Execution time: 4 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"I will load the numpy file from the specified working directory, parse the nested experiment_data structure, and compute final or best metrics as requested. For training and test, I will report the final metrics saved in the structure. For validation, I will compute the best values across epochs for accuracy, F1 score, and top-3 accuracy, and the minimum validation loss using the recorded loss history. The script prints the dataset name followed by clearly labeled metrics and executes immediately.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef safe_get(d, *keys, default=None):\n    cur = d\n    for k in keys:\n        if isinstance(cur, dict) and k in cur:\n            cur = cur[k]\n        else:\n            return default\n    return cur\n\n\ndef load_experiment_data():\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    path = os.path.join(working_dir, \"experiment_data.npy\")\n    if not os.path.isfile(path):\n        raise FileNotFoundError(f\"File not found: {path}\")\n    data = np.load(path, allow_pickle=True).item()\n    if not isinstance(data, dict):\n        raise ValueError(\"Loaded experiment_data is not a dict.\")\n    return data\n\n\ndef extract_final_train_metrics(ds_data):\n    # Expect a 'final' entry\n    train_entries = safe_get(ds_data, \"metrics\", \"train\", default=[])\n    final = None\n    for tag, vals in train_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"train loss\"] = final.get(\"loss\", None)\n        out[\"train accuracy\"] = final.get(\"acc\", None)\n        out[\"train F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"train top-3 accuracy\"] = final.get(\"top3\", None)\n    else:\n        # Fallback: try last value from losses for loss only\n        train_losses = safe_get(ds_data, \"losses\", \"train\", default=[])\n        if train_losses:\n            out[\"train loss\"] = train_losses[-1][1]\n    return out\n\n\ndef extract_best_val_metrics(ds_data):\n    # Collect per-epoch validation metrics (acc, macro_f1, top3)\n    val_entries = safe_get(ds_data, \"metrics\", \"val\", default=[])\n    best = {\n        \"validation accuracy\": None,\n        \"validation F1 score\": None,\n        \"validation top-3 accuracy\": None,\n    }\n    for tag, vals in val_entries:\n        if isinstance(vals, dict):\n            acc = vals.get(\"acc\", None)\n            f1 = vals.get(\"macro_f1\", None)\n            top3 = vals.get(\"top3\", None)\n            if acc is not None:\n                best[\"validation accuracy\"] = (\n                    acc\n                    if best[\"validation accuracy\"] is None\n                    else max(best[\"validation accuracy\"], acc)\n                )\n            if f1 is not None:\n                best[\"validation F1 score\"] = (\n                    f1\n                    if best[\"validation F1 score\"] is None\n                    else max(best[\"validation F1 score\"], f1)\n                )\n            if top3 is not None:\n                best[\"validation top-3 accuracy\"] = (\n                    top3\n                    if best[\"validation top-3 accuracy\"] is None\n                    else max(best[\"validation top-3 accuracy\"], top3)\n                )\n    # For validation loss, use recorded losses per epoch and take the minimum\n    val_losses = safe_get(ds_data, \"losses\", \"val\", default=[])\n    if val_losses:\n        min_val_loss = min((v for (_, v) in val_losses), default=None)\n    else:\n        # fallback: check 'final' val entry if present\n        min_val_loss = None\n        for tag, vals in val_entries:\n            if tag == \"final\" and isinstance(vals, dict):\n                min_val_loss = vals.get(\"loss\", None)\n                break\n    out = {\"validation loss\": min_val_loss}\n    out.update(best)\n    return out\n\n\ndef extract_final_test_metrics(ds_data):\n    test_entries = safe_get(ds_data, \"metrics\", \"test\", default=[])\n    final = None\n    for tag, vals in test_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"test loss\"] = final.get(\"loss\", None)\n        out[\"test accuracy\"] = final.get(\"acc\", None)\n        out[\"test F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"test top-3 accuracy\"] = final.get(\"top3\", None)\n    return out\n\n\ndef format_print_metrics(dataset_name, metrics_dict):\n    print(dataset_name)\n    # Keep a consistent order for readability\n    ordered_keys = [\n        \"train loss\",\n        \"train accuracy\",\n        \"train F1 score\",\n        \"train top-3 accuracy\",\n        \"validation loss\",\n        \"validation accuracy\",\n        \"validation F1 score\",\n        \"validation top-3 accuracy\",\n        \"test loss\",\n        \"test accuracy\",\n        \"test F1 score\",\n        \"test top-3 accuracy\",\n    ]\n    printed_any = False\n    for k in ordered_keys:\n        if k in metrics_dict and metrics_dict[k] is not None:\n            val = metrics_dict[k]\n            if isinstance(val, float):\n                print(f\"{k}: {val:.4f}\")\n            else:\n                print(f\"{k}: {val}\")\n            printed_any = True\n    # Print any remaining keys that were not in the ordered list\n    for k, v in metrics_dict.items():\n        if k not in ordered_keys and v is not None:\n            if isinstance(v, float):\n                print(f\"{k}: {v:.4f}\")\n            else:\n                print(f\"{k}: {v}\")\n            printed_any = True\n    if not printed_any:\n        print(\"No metrics available\")\n\n\ndef run():\n    experiment_data = load_experiment_data()\n    for ds_name, ds_data in experiment_data.items():\n        all_metrics = {}\n        all_metrics.update(extract_final_train_metrics(ds_data))\n        all_metrics.update(extract_best_val_metrics(ds_data))\n        all_metrics.update(extract_final_test_metrics(ds_data))\n        format_print_metrics(ds_name, all_metrics)\n\n\n# Execute immediately\nrun()\n","parse_term_out":["BPI2012","\n","train loss: 0.5148","\n","train accuracy: 0.7777","\n","train F1 score: 0.5609","\n","train top-3 accuracy: 0.9868","\n","validation loss: 0.5073","\n","validation accuracy: 0.7639","\n","validation F1 score: 0.6007","\n","validation top-3 accuracy: 0.9861","\n","test loss: 0.5355","\n","test accuracy: 0.7569","\n","test F1 score: 0.5872","\n","test top-3 accuracy: 0.9874","\n","BPI2017","\n","train loss: 0.3607","\n","train accuracy: 0.8422","\n","train F1 score: 0.5721","\n","train top-3 accuracy: 0.9941","\n","validation loss: 0.3756","\n","validation accuracy: 0.8405","\n","validation F1 score: 0.6180","\n","validation top-3 accuracy: 0.9928","\n","test loss: 0.3877","\n","test accuracy: 0.8332","\n","test F1 score: 0.5710","\n","test top-3 accuracy: 0.9906","\n","ROAD","\n","train loss: 0.4662","\n","train accuracy: 0.7894","\n","train F1 score: 0.5395","\n","train top-3 accuracy: 0.9986","\n","validation loss: 0.4274","\n","validation accuracy: 0.8122","\n","validation F1 score: 0.6664","\n","validation top-3 accuracy: 0.9969","\n","test loss: 0.4833","\n","test accuracy: 0.8020","\n","test F1 score: 0.4740","\n","test top-3 accuracy: 0.9936","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":240.2498710155487,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"Execution succeeded on all three XES logs with a working next-activity LSTM baseline, time-based case split, and metrics (accuracy, macro-F1, top-3). However, there is a subtle data leakage/double-normalization bug in feature preprocessing: build_prefix_dataset normalizes the time features (deltas, since_start) across all samples before the time-based split, and later the training function recomputes normalization statistics on the train subset and re-normalizes all splits again. This applies normalization twice and leaks global information into the training-set stats used later. Fix: remove any normalization inside build_prefix_dataset and keep raw features there. After creating the time-based split, compute means/stds exclusively on the training samples\u2019 raw features, then apply that single normalization to train/val/test. Optionally, rebuild the activity vocabulary act2id strictly from the training set to avoid leaking knowledge of unseen symbols; remap/OOv-handle activities in val/test. Additional robustness tweaks: set CrossEntropyLoss(ignore_index=pad_idx) (even though targets never use PAD), and consider masking the LSTM (or right-padding) so padding does not influence the recurrent dynamics.","exp_results_dir":null,"metric":{"value":{"metric_names":[{"metric_name":"loss","lower_is_better":true,"description":"Cross-entropy loss on the specified dataset split; lower values indicate better performance.","data":[{"dataset_name":"BPI2012","final_value":0.5355,"best_value":0.5073},{"dataset_name":"BPI2017","final_value":0.3877,"best_value":0.3607},{"dataset_name":"ROAD","final_value":0.4833,"best_value":0.4274}]},{"metric_name":"accuracy","lower_is_better":false,"description":"Top-1 accuracy on the specified dataset split; higher values indicate better performance.","data":[{"dataset_name":"BPI2012","final_value":0.7569,"best_value":0.7777},{"dataset_name":"BPI2017","final_value":0.8332,"best_value":0.8422},{"dataset_name":"ROAD","final_value":0.802,"best_value":0.8122}]},{"metric_name":"F1 score","lower_is_better":false,"description":"F1 score (harmonic mean of precision and recall) on the specified dataset split; higher values indicate better performance.","data":[{"dataset_name":"BPI2012","final_value":0.5872,"best_value":0.6007},{"dataset_name":"BPI2017","final_value":0.571,"best_value":0.618},{"dataset_name":"ROAD","final_value":0.474,"best_value":0.6664}]},{"metric_name":"top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the specified dataset split; higher values indicate better performance.","data":[{"dataset_name":"BPI2012","final_value":0.9874,"best_value":0.9874},{"dataset_name":"BPI2017","final_value":0.9906,"best_value":0.9941},{"dataset_name":"ROAD","final_value":0.9936,"best_value":0.9986}]}]},"maximize":null,"name":null,"description":null},"is_buggy":true,"is_buggy_plots":null,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":[],"plot_paths":[],"plot_analyses":[],"vlm_feedback_summary":[],"datasets_successfully_tested":[],"ablation_name":null,"hyperparam_name":null,"is_seed_node":true,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"# Set random seed\nimport random\nimport numpy as np\nimport torch\n\nseed = 2\nrandom.seed(seed)\nnp.random.seed(seed)\ntorch.manual_seed(seed)\nif torch.cuda.is_available():\n    torch.cuda.manual_seed(seed)\n\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport numpy as np\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import f1_score, accuracy_score\nimport random\nimport math\nimport matplotlib.pyplot as plt\n\n# Device handling (required)\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n# Experiment data structure\nexperiment_data = {\n    \"BPI2012\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"BPI2017\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n    \"ROAD\": {\n        \"metrics\": {\"train\": [], \"val\": [], \"test\": []},\n        \"losses\": {\"train\": [], \"val\": []},\n        \"predictions\": [],\n        \"ground_truth\": [],\n        \"epochs\": [],\n    },\n}\n\n# Data loading utilities (use provided helper)\nfrom ai_scientist.ideas.my_research_topic import load_datasets, pick_default_dataset\n\n\n# Reproducibility\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# Build prefixes\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    # Keep only 'complete' transitions if lifecycle exists\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        df = df[df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")]\n        if len(df) == 0:\n            df = df.copy()  # fallback if empty\n            df = df.sort_values([\"case_id\", \"timestamp\"])\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    # Build activity vocab\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}  # 0 for PAD\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        # Convert to numpy arrays for safe positional indexing\n        ts_ns = (\n            pd.to_datetime(g[\"timestamp\"], utc=True).astype(\"int64\").to_numpy()\n        )  # nanoseconds\n        ts = (ts_ns // 10**9).astype(np.int64)  # seconds as numpy array\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str).tolist()], dtype=np.int64\n        )\n        # simple calendar features\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        # time deltas and since start in seconds\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(\n            np.float32\n        )  # [T,5]\n        T = len(acts_ids)\n        if T < 2:\n            continue\n        # Generate prefixes of length k (min_prefix_len..min(max_prefix_len, T-1)); target = activity at position k\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            seq_acts = acts_ids[:k].tolist()\n            seq_feats = feats[:k]\n            target = int(acts_ids[k])\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": seq_acts,\n                    \"seq_feats\": seq_feats.copy(),\n                    \"target\": target,\n                    \"last_ts\": int(ts[k - 1]),\n                    \"next_ts\": int(ts[k]),\n                }\n            )\n\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n\n    # Collect feature normalization stats over all feats (initial; will be recomputed on train split)\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n    ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n    for s in samples:\n        if s[\"seq_feats\"].shape[0] > 0:\n            s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n            s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n    return samples, act2id, id2act, pad_id\n\n\n# Time-based split by case start time\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.zeros((pad_len, self.num_cont), dtype=np.float32)\n        feats_pad = np.vstack([feats_pad, feats.astype(np.float32)])\n        attn_mask = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad, dtype=torch.long),\n            \"feats\": torch.tensor(feats_pad, dtype=torch.float32),\n            \"mask\": torch.tensor(attn_mask, dtype=torch.float32),\n            \"y\": torch.tensor(s[\"target\"], dtype=torch.long),\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(\n        self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, num_layers=1, pad_idx=0\n    ):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim,\n            hidden_size=hidden,\n            batch_first=True,\n            num_layers=num_layers,\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)  # includes PAD index\n        self.pad_idx = pad_idx\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)  # [B,T,emb]\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h_last = h[-1]\n        h_last = self.dropout(h_last)\n        logits = self.fc(h_last)\n        return logits\n\n\ndef collate_fn(batch):\n    keys = batch[0].keys()\n    out = {k: torch.stack([b[k] for b in batch], dim=0) for k in keys}\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys, preds_top1, preds_probs = [], [], []\n    top3_correct = 0\n    n_total = 0\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk_idx = torch.topk(probs, k=k_val, dim=1)\n            ys.extend(batch[\"y\"].detach().cpu().tolist())\n            preds_top1.extend(top1.detach().cpu().tolist())\n            preds_probs.append(probs.detach().cpu().numpy())\n            # top-3 correctness\n            for i in range(batch[\"y\"].size(0)):\n                if batch[\"y\"][i].item() in topk_idx[i].detach().cpu().tolist():\n                    top3_correct += 1\n            n_total += batch[\"y\"].size(0)\n    avg_loss = total_loss / max(1, n_total)\n    y_true = np.array(ys)\n    y_pred = np.array(preds_top1)\n    mask = y_true != pad_idx\n    y_true = y_true[mask]\n    y_pred = y_pred[mask]\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except Exception:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n_total))\n    probs_concat = (\n        np.concatenate(preds_probs, axis=0)\n        if len(preds_probs) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return avg_loss, acc, f1, top3, y_true, y_pred, probs_concat\n\n\ndef train_one_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    # Time-based split\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    # Build samples across all to get vocab; we'll re-normalize with train stats\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    # Filter per split\n    samples_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    samples_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    samples_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # Recompute normalization using train samples only\n    if len(samples_train) > 0:\n        concat_feats = [\n            s[\"seq_feats\"] for s in samples_train if s[\"seq_feats\"].shape[0] > 0\n        ]\n        if len(concat_feats) > 0:\n            all_feats = np.concatenate(concat_feats, axis=0)\n            dt_mean, dt_std = all_feats[:, 0].mean(), all_feats[:, 0].std() + 1e-6\n            ss_mean, ss_std = all_feats[:, 1].mean(), all_feats[:, 1].std() + 1e-6\n\n            def norm_samples(samples):\n                for s in samples:\n                    if s[\"seq_feats\"].shape[0] > 0:\n                        s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                        s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n            norm_samples(samples_train)\n            norm_samples(samples_val)\n            norm_samples(samples_test)\n    print(\n        f\"Samples train/val/test: {len(samples_train)}/{len(samples_val)}/{len(samples_test)}; vocab={len(act2id)}\"\n    )\n    if len(samples_train) == 0 or len(act2id) < 2:\n        print(\"Not enough data to train. Skipping.\")\n        return\n    ds_train = PrefixDataset(\n        samples_train, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_val = PrefixDataset(\n        samples_val, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    ds_test = PrefixDataset(\n        samples_test, pad_id=pad_id, max_len=max_prefix_len, num_cont=5\n    )\n    dl_train = DataLoader(\n        ds_train,\n        batch_size=batch_size,\n        shuffle=True,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_val = DataLoader(\n        ds_val,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n    dl_test = DataLoader(\n        ds_test,\n        batch_size=batch_size,\n        shuffle=False,\n        collate_fn=collate_fn,\n        num_workers=0,\n    )\n\n    # Model\n    model = LSTMBaseline(\n        vocab_size=len(act2id),\n        emb_dim=64,\n        cont_dim=5,\n        hidden=128,\n        num_layers=1,\n        pad_idx=pad_id,\n    ).to(device)\n    criterion = nn.CrossEntropyLoss().to(device)\n    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n\n    # Training loop\n    best_val_top3 = -1.0\n    best_state = None\n    hist = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for epoch in range(1, max_epochs + 1):\n        model.train()\n        total = 0\n        running_loss = 0.0\n        for batch in dl_train:\n            batch = {\n                k: v.to(device) for k, v in batch.items() if isinstance(v, torch.Tensor)\n            }\n            optimizer.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            loss.backward()\n            optimizer.step()\n            running_loss += loss.item() * logits.size(0)\n            total += logits.size(0)\n        train_loss = running_loss / max(1, total)\n        val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n            model, dl_val, criterion, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {epoch}: validation_loss = {val_loss:.4f} | val_acc={val_acc:.4f} | val_f1={val_f1:.4f} | val_top3={val_top3:.4f}\"\n        )\n        hist[\"train_loss\"].append(train_loss)\n        hist[\"val_loss\"].append(val_loss)\n        hist[\"val_top3\"].append(val_top3)\n        experiment_data[name][\"losses\"][\"train\"].append((epoch, train_loss))\n        experiment_data[name][\"losses\"][\"val\"].append((epoch, val_loss))\n        experiment_data[name][\"metrics\"][\"val\"].append(\n            (epoch, {\"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3})\n        )\n        experiment_data[name][\"epochs\"].append(epoch)\n        if val_top3 > best_val_top3:\n            best_val_top3 = val_top3\n            best_state = {k: v.cpu().clone() for k, v in model.state_dict().items()}\n\n    # Load best\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n\n    # Final eval on train/val/test\n    train_loss, train_acc, train_f1, train_top3, _, _, _ = evaluate(\n        model, dl_train, criterion, device, len(act2id), pad_id\n    )\n    val_loss, val_acc, val_f1, val_top3, _, _, _ = evaluate(\n        model, dl_val, criterion, device, len(act2id), pad_id\n    )\n    test_loss, test_acc, test_f1, test_top3, y_true_t, y_pred_t, probs_t = evaluate(\n        model, dl_test, criterion, device, len(act2id), pad_id\n    )\n    print(\n        f\"[{name}] Train: loss={train_loss:.4f} acc={train_acc:.4f} f1={train_f1:.4f} top3={train_top3:.4f}\"\n    )\n    print(\n        f\"[{name}] Test:  loss={test_loss:.4f} acc={test_acc:.4f} f1={test_f1:.4f} top3={test_top3:.4f}\"\n    )\n\n    # Save metrics\n    experiment_data[name][\"metrics\"][\"train\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": train_loss,\n                \"acc\": train_acc,\n                \"macro_f1\": train_f1,\n                \"top3\": train_top3,\n            },\n        )\n    )\n    experiment_data[name][\"metrics\"][\"val\"].append(\n        (\n            \"final\",\n            {\"loss\": val_loss, \"acc\": val_acc, \"macro_f1\": val_f1, \"top3\": val_top3},\n        )\n    )\n    experiment_data[name][\"metrics\"][\"test\"].append(\n        (\n            \"final\",\n            {\n                \"loss\": test_loss,\n                \"acc\": test_acc,\n                \"macro_f1\": test_f1,\n                \"top3\": test_top3,\n            },\n        )\n    )\n    experiment_data[name][\"predictions\"] = y_pred_t.tolist()\n    experiment_data[name][\"ground_truth\"] = y_true_t.tolist()\n\n    # Plots\n    try:\n        plt.figure()\n        plt.plot(hist[\"train_loss\"], label=\"train_loss\")\n        plt.plot(hist[\"val_loss\"], label=\"val_loss\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"loss_curves_{name}.png\"))\n        plt.close()\n\n        plt.figure()\n        plt.plot(hist[\"val_top3\"], label=\"val_top3\")\n        plt.legend()\n        plt.title(f\"Val Top-3 Acc - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Top-3 Acc\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"val_top3_{name}.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"[warn] Plotting failed: {e}\")\n\n    # Save confusion matrix-like data (optional)\n    try:\n        from sklearn.metrics import confusion_matrix\n\n        cm = confusion_matrix(y_true_t, y_pred_t)\n        np.save(os.path.join(working_dir, f\"cm_{name}.npy\"), cm)\n    except Exception as e:\n        print(f\"[warn] Confusion matrix failed: {e}\")\n\n\ndef main():\n    datasets = load_datasets()\n    # Loop through loaded datasets; cap to 5000 earliest cases for speed\n    for key, df in datasets.items():\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep_cases = set(starts.iloc[:5000][\"case_id\"])\n                df_small = df[df[\"case_id\"].isin(keep_cases)].copy()\n            else:\n                df_small = df\n        except Exception:\n            df_small = df\n        train_one_dataset(\n            key, df_small, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    np.savez_compressed(\n        os.path.join(working_dir, \"experiment_data_compressed.npz\"),\n        data=experiment_data,\n    )\n\n\n# Execute immediately\nmain()\n","plan":"Seed node","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\nimport pandas as pd\nimport torch\nfrom torch import nn\nfrom torch.utils.data import Dataset, DataLoader\nfrom sklearn.metrics import (\n    f1_score,\n    accuracy_score,\n    confusion_matrix,\n    precision_recall_curve,\n    average_precision_score,\n)\nfrom sklearn.preprocessing import label_binarize\nimport random\nimport math\nfrom collections import defaultdict\n\n# Reproducibility and device\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\nprint(f\"Using device: {device}\")\n\n\ndef set_seed(seed=42):\n    random.seed(seed)\n    np.random.seed(seed)\n    torch.manual_seed(seed)\n    if torch.cuda.is_available():\n        torch.cuda.manual_seed_all(seed)\n\n\nset_seed(42)\n\n\n# --------- Data loading from local XES via pm4py ----------\ndef load_xes_folder(data_dir=\"data\"):\n    datasets = {}\n    try:\n        import pm4py\n    except Exception as e:\n        print(f\"pm4py not available: {e}\")\n        return datasets\n    if not os.path.isdir(data_dir):\n        print(f\"Data directory not found: {data_dir}\")\n        return datasets\n    for fn in os.listdir(data_dir):\n        if fn.lower().endswith(\".xes\") or fn.lower().endswith(\".xes.gz\"):\n            path = os.path.join(data_dir, fn)\n            try:\n                log = pm4py.read_xes(path)\n                df = pm4py.convert_to_dataframe(log)\n                # Standardize columns\n                # pm4py dataframe typically has case:concept:name, concept:name, time:timestamp, lifecycle:transition\n                cols = df.columns\n                case_col = (\n                    \"case:concept:name\"\n                    if \"case:concept:name\" in cols\n                    else (\"case\" if \"case\" in cols else None)\n                )\n                act_col = (\n                    \"concept:name\"\n                    if \"concept:name\" in cols\n                    else (\"activity\" if \"activity\" in cols else None)\n                )\n                ts_col = (\n                    \"time:timestamp\"\n                    if \"time:timestamp\" in cols\n                    else (\"timestamp\" if \"timestamp\" in cols else None)\n                )\n                life_col = (\n                    \"lifecycle:transition\"\n                    if \"lifecycle:transition\" in cols\n                    else (\"lifecycle\" if \"lifecycle\" in cols else None)\n                )\n                if case_col is None or act_col is None or ts_col is None:\n                    print(f\"Missing required columns in {fn}, skipping.\")\n                    continue\n                out = pd.DataFrame(\n                    {\n                        \"case_id\": df[case_col].astype(str).values,\n                        \"activity\": df[act_col].astype(str).values,\n                        \"timestamp\": pd.to_datetime(df[ts_col], utc=True),\n                    }\n                )\n                if life_col is not None:\n                    out[\"lifecycle\"] = df[life_col].astype(str).values\n                name = os.path.splitext(fn)[0]\n                datasets[name] = out\n                print(\n                    f\"Loaded {name}: {len(out)} events, {out['case_id'].nunique()} cases\"\n                )\n            except Exception as e:\n                print(f\"Failed to load {fn}: {e}\")\n    return datasets\n\n\n# --------- Prefix building and split ----------\ndef build_prefix_dataset(df, max_prefix_len=10, min_prefix_len=1):\n    df = df.copy()\n    if \"lifecycle\" in df.columns:\n        mask = df[\"lifecycle\"].astype(str).str.lower().eq(\"complete\")\n        if mask.any():\n            df = df[mask]\n    df = df.sort_values([\"case_id\", \"timestamp\"])\n    acts = df[\"activity\"].astype(str).unique().tolist()\n    act2id = {a: i + 1 for i, a in enumerate(sorted(acts))}\n    id2act = {i: a for a, i in act2id.items()}\n    pad_id = 0\n    samples = []\n    for cid, g in df.groupby(\"case_id\"):\n        g = g.sort_values(\"timestamp\")\n        if len(g) < 2:\n            continue\n        g_ts = pd.to_datetime(g[\"timestamp\"], utc=True)\n        ts = (g_ts.astype(\"int64\") // 10**9).to_numpy(np.int64)\n        acts_ids = np.array(\n            [act2id[a] for a in g[\"activity\"].astype(str)], dtype=np.int64\n        )\n        hours = (g_ts.dt.hour.to_numpy(dtype=float) / 23.0).astype(np.float32)\n        weekdays = (g_ts.dt.weekday.to_numpy(dtype=float) / 6.0).astype(np.float32)\n        working = (\n            (g_ts.dt.weekday.to_numpy() < 5)\n            & (g_ts.dt.hour.to_numpy() >= 8)\n            & (g_ts.dt.hour.to_numpy() <= 17)\n        ).astype(np.float32)\n        deltas = np.diff(ts, prepend=ts[0]).astype(np.float32)\n        since_start = (ts - ts[0]).astype(np.float32)\n        feats = np.stack(\n            [deltas, since_start, hours, weekdays, working], axis=1\n        ).astype(np.float32)\n        T = len(acts_ids)\n        max_k = min(max_prefix_len, T - 1)\n        for k in range(min_prefix_len, max_k + 1):\n            samples.append(\n                {\n                    \"case_id\": cid,\n                    \"seq_acts\": acts_ids[:k].tolist(),\n                    \"seq_feats\": feats[:k].copy(),\n                    \"target\": int(acts_ids[k]),\n                    \"prefix_len\": k,\n                }\n            )\n    if len(samples) == 0:\n        return samples, act2id, id2act, pad_id\n    all_feats = np.concatenate(\n        [s[\"seq_feats\"] for s in samples if len(s[\"seq_feats\"]) > 0], axis=0\n    )\n    for s in samples:\n        pass  # initial no norm; will norm on train split\n    return samples, act2id, id2act, pad_id\n\n\ndef time_based_split(df, train_frac=0.7, val_frac=0.15):\n    starts = (\n        df.sort_values(\"timestamp\").groupby(\"case_id\")[\"timestamp\"].min().reset_index()\n    )\n    starts = starts.sort_values(\"timestamp\").reset_index(drop=True)\n    n = len(starts)\n    n_train = int(n * train_frac)\n    n_val = int(n * val_frac)\n    train_cases = set(starts.iloc[:n_train][\"case_id\"])\n    val_cases = set(starts.iloc[n_train : n_train + n_val][\"case_id\"])\n    test_cases = set(starts.iloc[n_train + n_val :][\"case_id\"])\n    return train_cases, val_cases, test_cases\n\n\nclass PrefixDataset(Dataset):\n    def __init__(self, samples, pad_id, max_len=10, num_cont=5):\n        self.samples = samples\n        self.pad_id = pad_id\n        self.max_len = max_len\n        self.num_cont = num_cont\n\n    def __len__(self):\n        return len(self.samples)\n\n    def __getitem__(self, idx):\n        s = self.samples[idx]\n        seq = s[\"seq_acts\"][-self.max_len :]\n        feats = s[\"seq_feats\"][-self.max_len :]\n        L = len(seq)\n        pad_len = self.max_len - L\n        seq_pad = [self.pad_id] * pad_len + seq\n        feats_pad = np.vstack(\n            [\n                np.zeros((pad_len, self.num_cont), dtype=np.float32),\n                feats.astype(np.float32),\n            ]\n        )\n        attn = np.array([0] * pad_len + [1] * L, dtype=np.float32)\n        return {\n            \"acts\": torch.tensor(seq_pad).long(),\n            \"feats\": torch.tensor(feats_pad).float(),\n            \"mask\": torch.tensor(attn).float(),\n            \"y\": torch.tensor(s[\"target\"]).long(),\n            \"prefix_len\": L,\n        }\n\n\nclass LSTMBaseline(nn.Module):\n    def __init__(self, vocab_size, emb_dim=64, cont_dim=5, hidden=128, pad_idx=0):\n        super().__init__()\n        self.emb = nn.Embedding(vocab_size + 1, emb_dim, padding_idx=pad_idx)\n        self.lstm = nn.LSTM(\n            input_size=emb_dim + cont_dim, hidden_size=hidden, batch_first=True\n        )\n        self.dropout = nn.Dropout(0.2)\n        self.fc = nn.Linear(hidden, vocab_size + 1)\n\n    def forward(self, acts, feats, mask):\n        x = self.emb(acts)\n        x = torch.cat([x, feats], dim=-1)\n        out, (h, c) = self.lstm(x)\n        h = self.dropout(h[-1])\n        return self.fc(h)\n\n\ndef collate_fn(batch):\n    out = {\n        k: (\n            torch.stack([b[k] for b in batch], 0)\n            if isinstance(batch[0][k], torch.Tensor)\n            else [b[k] for b in batch]\n        )\n        for k in batch[0].keys()\n    }\n    return out\n\n\ndef evaluate(model, loader, criterion, device, num_classes, pad_idx):\n    model.eval()\n    total_loss = 0.0\n    ys = []\n    yhat = []\n    probs_list = []\n    n = 0\n    top3_correct = 0\n    pref_lens = []\n    top3_flags = []\n    with torch.no_grad():\n        for batch in loader:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = criterion(logits, batch[\"y\"])\n            total_loss += loss.item() * logits.size(0)\n            probs = torch.softmax(logits, dim=1)\n            top1 = torch.argmax(probs, dim=1)\n            k_val = min(3, probs.size(1))\n            _, topk = torch.topk(probs, k=k_val, dim=1)\n            y = batch[\"y\"]\n            ys.extend(y.detach().cpu().tolist())\n            yhat.extend(top1.detach().cpu().tolist())\n            probs_list.append(probs.detach().cpu().numpy())\n            for i in range(y.size(0)):\n                flag = int(y[i].item() in topk[i].detach().cpu().tolist())\n                top3_correct += flag\n                top3_flags.append(flag)\n                pref_lens.append(int(batch[\"prefix_len\"][i].item()))\n            n += y.size(0)\n    avg_loss = total_loss / max(1, n)\n    y_true = np.array(ys)\n    y_pred = np.array(yhat)\n    acc = float(accuracy_score(y_true, y_pred)) if len(y_true) > 0 else 0.0\n    try:\n        f1 = float(f1_score(y_true, y_pred, average=\"macro\"))\n    except:\n        f1 = 0.0\n    top3 = float(top3_correct / max(1, n))\n    probs_concat = (\n        np.concatenate(probs_list, axis=0)\n        if len(probs_list) > 0\n        else np.zeros((0, num_classes + 1))\n    )\n    return (\n        avg_loss,\n        acc,\n        f1,\n        top3,\n        y_true,\n        y_pred,\n        probs_concat,\n        np.array(pref_lens),\n        np.array(top3_flags),\n    )\n\n\ndef train_on_dataset(\n    name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n):\n    print(f\"\\n=== Dataset: {name} ===\")\n    train_cases, val_cases, test_cases = time_based_split(df, 0.7, 0.15)\n    samples_all, act2id, id2act, pad_id = build_prefix_dataset(\n        df, max_prefix_len=max_prefix_len\n    )\n    s_train = [s for s in samples_all if s[\"case_id\"] in train_cases]\n    s_val = [s for s in samples_all if s[\"case_id\"] in val_cases]\n    s_test = [s for s in samples_all if s[\"case_id\"] in test_cases]\n    # normalize time features on train\n    if len(s_train) > 0:\n        feats = np.concatenate(\n            [s[\"seq_feats\"] for s in s_train if len(s[\"seq_feats\"]) > 0], axis=0\n        )\n        dt_mean, dt_std = feats[:, 0].mean(), feats[:, 0].std() + 1e-6\n        ss_mean, ss_std = feats[:, 1].mean(), feats[:, 1].std() + 1e-6\n\n        def norm(samples):\n            for s in samples:\n                if s[\"seq_feats\"].shape[0] > 0:\n                    s[\"seq_feats\"][:, 0] = (s[\"seq_feats\"][:, 0] - dt_mean) / dt_std\n                    s[\"seq_feats\"][:, 1] = (s[\"seq_feats\"][:, 1] - ss_mean) / ss_std\n\n        norm(s_train)\n        norm(s_val)\n        norm(s_test)\n    print(\n        f\"Samples train/val/test: {len(s_train)}/{len(s_val)}/{len(s_test)}; vocab={len(act2id)}\"\n    )\n    if len(s_train) == 0 or len(act2id) < 2:\n        print(\"Insufficient data; skipping.\")\n        return None\n    ds_tr = PrefixDataset(s_train, pad_id, max_prefix_len, 5)\n    ds_va = PrefixDataset(s_val, pad_id, max_prefix_len, 5)\n    ds_te = PrefixDataset(s_test, pad_id, max_prefix_len, 5)\n    dl_tr = DataLoader(\n        ds_tr, batch_size=batch_size, shuffle=True, collate_fn=collate_fn\n    )\n    dl_va = DataLoader(\n        ds_va, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    dl_te = DataLoader(\n        ds_te, batch_size=batch_size, shuffle=False, collate_fn=collate_fn\n    )\n    model = LSTMBaseline(\n        vocab_size=len(act2id), emb_dim=64, cont_dim=5, hidden=128, pad_idx=pad_id\n    ).to(device)\n    crit = nn.CrossEntropyLoss().to(device)\n    opt = torch.optim.Adam(model.parameters(), lr=lr)\n    best_top3 = -1.0\n    best_state = None\n    history = {\"train_loss\": [], \"val_loss\": [], \"val_top3\": []}\n    for ep in range(1, max_epochs + 1):\n        model.train()\n        tot = 0\n        run_loss = 0.0\n        for batch in dl_tr:\n            batch = {\n                k: v.to(device) if isinstance(v, torch.Tensor) else v\n                for k, v in batch.items()\n            }\n            opt.zero_grad()\n            logits = model(batch[\"acts\"], batch[\"feats\"], batch[\"mask\"])\n            loss = crit(logits, batch[\"y\"])\n            loss.backward()\n            opt.step()\n            run_loss += loss.item() * logits.size(0)\n            tot += logits.size(0)\n        tr_loss = run_loss / max(1, tot)\n        va_loss, va_acc, va_f1, va_top3, *_ = evaluate(\n            model, dl_va, crit, device, len(act2id), pad_id\n        )\n        print(\n            f\"Epoch {ep}: val_loss={va_loss:.4f} acc={va_acc:.4f} f1={va_f1:.4f} top3={va_top3:.4f}\"\n        )\n        history[\"train_loss\"].append(tr_loss)\n        history[\"val_loss\"].append(va_loss)\n        history[\"val_top3\"].append(va_top3)\n        if va_top3 > best_top3:\n            best_top3 = va_top3\n            best_state = {\n                k: v.detach().cpu().clone() for k, v in model.state_dict().items()\n            }\n    if best_state is not None:\n        model.load_state_dict(best_state)\n        model.to(device)\n    tr_loss, tr_acc, tr_f1, tr_top3, *_ = evaluate(\n        model, dl_tr, crit, device, len(act2id), pad_id\n    )\n    te_loss, te_acc, te_f1, te_top3, y_true, y_pred, probs, pref_lens, top3_flags = (\n        evaluate(model, dl_te, crit, device, len(act2id), pad_id)\n    )\n    print(\n        f\"[{name}] Test: loss={te_loss:.4f} acc={te_acc:.4f} f1={te_f1:.4f} top3={te_top3:.4f}\"\n    )\n    # package experiment data\n    exp = {\n        \"metrics\": {\n            \"train\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": tr_loss,\n                        \"acc\": tr_acc,\n                        \"macro_f1\": tr_f1,\n                        \"top3\": tr_top3,\n                    },\n                )\n            ],\n            \"val\": [],\n            \"test\": [\n                (\n                    \"final\",\n                    {\n                        \"loss\": te_loss,\n                        \"acc\": te_acc,\n                        \"macro_f1\": te_f1,\n                        \"top3\": te_top3,\n                    },\n                )\n            ],\n        },\n        \"losses\": {\n            \"train\": list(enumerate(history[\"train_loss\"], start=1)),\n            \"val\": list(enumerate(history[\"val_loss\"], start=1)),\n        },\n        \"predictions\": y_pred.tolist(),\n        \"ground_truth\": y_true.tolist(),\n        \"epochs\": list(range(1, len(history[\"train_loss\"]) + 1)),\n        \"probs\": probs,\n        \"prefix_lens\": pref_lens.tolist(),\n        \"top3_flags\": top3_flags.tolist(),\n        \"act2id\": act2id,\n    }\n    # plots for this dataset\n    try:\n        plt.figure()\n        plt.plot(history[\"train_loss\"], label=\"train\")\n        plt.plot(history[\"val_loss\"], label=\"val\")\n        plt.legend()\n        plt.title(f\"Loss Curves - {name}\")\n        plt.xlabel(\"Epoch\")\n        plt.ylabel(\"Loss\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating loss curves for {name}: {e}\")\n        plt.close()\n    try:\n        cm = confusion_matrix(y_true, y_pred)\n        plt.figure(figsize=(5, 4))\n        plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n        plt.colorbar()\n        plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n        plt.xlabel(\"Predicted\")\n        plt.ylabel(\"True\")\n        plt.tight_layout()\n        plt.savefig(os.path.join(working_dir, f\"{name}_confusion_matrix.png\"))\n        plt.close()\n    except Exception as e:\n        print(f\"Error creating confusion matrix for {name}: {e}\")\n        plt.close()\n    try:\n        # Top-3 accuracy vs prefix length\n        if len(pref_lens) > 0:\n            d = defaultdict(list)\n            for L, flag in zip(pref_lens, top3_flags):\n                d[int(L)].append(int(flag))\n            xs = sorted(d.keys())\n            ys = [np.mean(d[k]) for k in xs]\n            plt.figure()\n            plt.plot(xs, ys, marker=\"o\")\n            plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n            plt.xlabel(\"Prefix Length\")\n            plt.ylabel(\"Top-3 Accuracy\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_top3_vs_prefixlen.png\"))\n            plt.close()\n    except Exception as e:\n        print(f\"Error creating Top-3 vs prefix length for {name}: {e}\")\n        plt.close()\n    try:\n        # Macro PR curve (one-vs-rest); may be coarse due to many classes\n        if probs.shape[0] > 0:\n            classes = np.unique(y_true)\n            Y = label_binarize(y_true, classes=range(probs.shape[1]))\n            # only keep columns present in classes to avoid PAD\n            present = [c for c in classes]\n            if len(present) > 1:\n                precisions = []\n                recalls = []\n                aps = []\n                for c in present:\n                    p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                    ap = average_precision_score(Y[:, c], probs[:, c])\n                    precisions.append(\n                        np.interp(np.linspace(0, 1, 101), r[::-1], p[::-1])\n                    )\n                    recalls.append(np.linspace(0, 1, 101))\n                    aps.append(ap)\n                macro_p = np.mean(np.stack(precisions, 0), 0)\n                macro_r = np.linspace(0, 1, 101)\n                plt.figure()\n                plt.plot(macro_r, macro_p, label=f\"Macro-PR (mAP={np.mean(aps):.3f})\")\n                plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Recall\")\n                plt.ylabel(\"Precision\")\n                plt.legend()\n                plt.tight_layout()\n                plt.savefig(os.path.join(working_dir, f\"{name}_macro_pr.png\"))\n                plt.close()\n    except Exception as e:\n        print(f\"Error creating PR curve for {name}: {e}\")\n        plt.close()\n    return name, exp\n\n\ndef main():\n    datasets = load_xes_folder(data_dir=os.path.join(os.getcwd(), \"data\"))\n    experiment_data = {}\n    for name, df in datasets.items():\n        # optional cap earliest 5000 cases\n        try:\n            starts = (\n                df.sort_values(\"timestamp\")\n                .groupby(\"case_id\")[\"timestamp\"]\n                .min()\n                .reset_index()\n            )\n            if len(starts) > 5000:\n                keep = set(starts.iloc[:5000][\"case_id\"])\n                df = df[df[\"case_id\"].isin(keep)].copy()\n        except:\n            pass\n        res = train_on_dataset(\n            name, df, max_epochs=10, batch_size=128, max_prefix_len=10, lr=1e-3\n        )\n        if res is not None:\n            k, exp = res\n            experiment_data[k] = exp\n    # Save experiment data\n    np.save(os.path.join(working_dir, \"experiment_data.npy\"), experiment_data)\n    # Print evaluation metrics\n    for k, v in experiment_data.items():\n        test_metrics = dict(v[\"metrics\"][\"test\"][0][1])\n        print(\n            f\"{k} | Test acc={test_metrics['acc']:.4f} macro_f1={test_metrics['macro_f1']:.4f} top3={test_metrics['top3']:.4f} loss={test_metrics['loss']:.4f}\"\n        )\n\n    # Secondary plotting pass strictly from experiment_data.npy (as required)\n    try:\n        experiment_data_loaded = np.load(\n            os.path.join(working_dir, \"experiment_data.npy\"), allow_pickle=True\n        ).item()\n    except Exception as e:\n        print(f\"Error loading experiment data: {e}\")\n        experiment_data_loaded = {}\n    for name, ed in experiment_data_loaded.items():\n        try:\n            # re-plot loss curves from saved data\n            plt.figure()\n            tl = [y for (_, y) in ed.get(\"losses\", {}).get(\"train\", [])]\n            vl = [y for (_, y) in ed.get(\"losses\", {}).get(\"val\", [])]\n            if len(tl) > 0:\n                plt.plot(tl, label=\"train\")\n            if len(vl) > 0:\n                plt.plot(vl, label=\"val\")\n            plt.legend()\n            plt.title(f\"Loss Curves - {name}\\nNext-activity\")\n            plt.xlabel(\"Epoch\")\n            plt.ylabel(\"Loss\")\n            plt.tight_layout()\n            plt.savefig(os.path.join(working_dir, f\"{name}_loss_curves_reload.png\"))\n            plt.close()\n        except Exception as e:\n            print(f\"Error creating plot1: {e}\")\n            plt.close()\n        try:\n            # confusion matrix from predictions and ground truth\n            y_true = ed.get(\"ground_truth\", [])\n            y_pred = ed.get(\"predictions\", [])\n            if len(y_true) > 0 and len(y_pred) > 0:\n                cm = confusion_matrix(y_true, y_pred)\n                plt.figure(figsize=(5, 4))\n                plt.imshow(cm, aspect=\"auto\", cmap=\"Blues\")\n                plt.colorbar()\n                plt.title(f\"Confusion Matrix (Test) - {name}\\nNext-activity\")\n                plt.xlabel(\"Predicted\")\n                plt.ylabel(\"True\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_confusion_matrix_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot2: {e}\")\n            plt.close()\n        try:\n            # Top-3 vs prefix length if present\n            pref = ed.get(\"prefix_lens\", [])\n            flags = ed.get(\"top3_flags\", [])\n            if len(pref) > 0 and len(flags) > 0:\n                d = defaultdict(list)\n                for L, f in zip(pref, flags):\n                    d[int(L)].append(int(f))\n                xs = sorted(d.keys())\n                ys = [float(np.mean(d[x])) for x in xs]\n                plt.figure()\n                plt.plot(xs, ys, marker=\"o\")\n                plt.title(f\"Top-3 Accuracy vs Prefix Length - {name}\\nNext-activity\")\n                plt.xlabel(\"Prefix Length\")\n                plt.ylabel(\"Top-3 Accuracy\")\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_top3_vs_prefixlen_reload.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating plot3: {e}\")\n            plt.close()\n        try:\n            # Macro PR curve if probs available\n            probs = np.array(ed.get(\"probs\", []))\n            y_true = ed.get(\"ground_truth\", [])\n            if probs.size > 0 and len(y_true) > 0:\n                classes = sorted(set(y_true))\n                Y = label_binarize(np.array(y_true), classes=range(probs.shape[1]))\n                present = classes\n                if len(present) > 1:\n                    precisions = []\n                    aps = []\n                    grid = np.linspace(0, 1, 101)\n                    for c in present:\n                        p, r, _ = precision_recall_curve(Y[:, c], probs[:, c])\n                        precisions.append(np.interp(grid, r[::-1], p[::-1]))\n                        aps.append(average_precision_score(Y[:, c], probs[:, c]))\n                    macro_p = np.mean(np.stack(precisions, 0), 0)\n                    plt.figure()\n                    plt.plot(grid, macro_p, label=f\"mAP={np.mean(aps):.3f}\")\n                    plt.title(f\"Macro Precision-Recall (Test) - {name}\\nNext-activity\")\n                    plt.xlabel(\"Recall\")\n                    plt.ylabel(\"Precision\")\n                    plt.legend()\n                    plt.tight_layout()\n                    plt.savefig(\n                        os.path.join(working_dir, f\"{name}_macro_pr_reload.png\")\n                    )\n                    plt.close()\n        except Exception as e:\n            print(f\"Error creating plot4: {e}\")\n            plt.close()\n\n\nif __name__ == \"__main__\":\n    main()\n","plot_plan":null,"step":8,"id":"74c75ff464094554b5c0e6b72036923c","ctime":1757758298.9856772,"_term_out":["Using device: cuda","\n","[data] Using discovered data dir: /workspace/data","\n","[data] Available in /workspace/data: ['BPI_Challenge_2012.xes', 'BPI_Challenge_2017.xes', 'Road_Traffic_Fine_Management_Process.xes']","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2012.xes","\n","\rparsing log, completed traces ::   0%|          | 0/13087 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/13087 [00:00<22:26,  9.72it/s]","\rparsing log, completed traces ::   1%|1         | 157/13087 [00:00<00:14, 907.01it/s]","\rparsing log, completed traces ::   2%|2         | 311/13087 [00:00<00:10, 1192.67it/s]","\rparsing log, completed traces ::   4%|3         | 478/13087 [00:00<00:09, 1377.15it/s]","\rparsing log, completed traces ::   5%|4         | 646/13087 [00:00<00:08, 1482.64it/s]","\rparsing log, completed traces ::   6%|6         | 830/13087 [00:00<00:07, 1600.45it/s]","\rparsing log, completed traces ::   8%|7         | 991/13087 [00:01<00:15, 773.14it/s] ","\rparsing log, completed traces ::   9%|8         | 1156/13087 [00:01<00:12, 931.99it/s]","\rparsing log, completed traces ::  10%|#         | 1325/13087 [00:01<00:10, 1087.60it/s]","\rparsing log, completed traces ::  12%|#1        | 1537/13087 [00:01<00:08, 1323.04it/s]","\rparsing log, completed traces ::  13%|#3        | 1703/13087 [00:01<00:08, 1400.36it/s]","\rparsing log, completed traces ::  14%|#4        | 1875/13087 [00:01<00:07, 1474.63it/s]","\rparsing log, completed traces ::  16%|#5        | 2041/13087 [00:01<00:07, 1512.24it/s]","\rparsing log, completed traces ::  17%|#6        | 2210/13087 [00:01<00:06, 1559.92it/s]","\rparsing log, completed traces ::  18%|#8        | 2392/13087 [00:01<00:06, 1631.74it/s]","\rparsing log, completed traces ::  20%|#9        | 2567/13087 [00:01<00:06, 1663.67it/s]","\rparsing log, completed traces ::  21%|##        | 2739/13087 [00:02<00:06, 1651.24it/s]","\rparsing log, completed traces ::  22%|##2       | 2908/13087 [00:02<00:06, 1623.01it/s]","\rparsing log, completed traces ::  23%|##3       | 3073/13087 [00:02<00:06, 1602.03it/s]","\rparsing log, completed traces ::  25%|##4       | 3271/13087 [00:02<00:05, 1709.60it/s]","\rparsing log, completed traces ::  27%|##6       | 3505/13087 [00:02<00:05, 1890.82it/s]","\rparsing log, completed traces ::  28%|##8       | 3696/13087 [00:02<00:05, 1854.48it/s]","\rparsing log, completed traces ::  30%|##9       | 3883/13087 [00:03<00:09, 954.06it/s] ","\rparsing log, completed traces ::  31%|###1      | 4117/13087 [00:03<00:07, 1197.56it/s]","\rparsing log, completed traces ::  33%|###3      | 4354/13087 [00:03<00:06, 1432.75it/s]","\rparsing log, completed traces ::  35%|###4      | 4579/13087 [00:03<00:05, 1614.63it/s]","\rparsing log, completed traces ::  37%|###6      | 4780/13087 [00:03<00:05, 1653.42it/s]","\rparsing log, completed traces ::  38%|###8      | 4979/13087 [00:03<00:04, 1735.02it/s]","\rparsing log, completed traces ::  40%|###9      | 5174/13087 [00:03<00:04, 1742.70it/s]","\rparsing log, completed traces ::  41%|####      | 5364/13087 [00:03<00:04, 1724.38it/s]","\rparsing log, completed traces ::  42%|####2     | 5547/13087 [00:03<00:04, 1664.04it/s]","\rparsing log, completed traces ::  44%|####3     | 5724/13087 [00:03<00:04, 1692.32it/s]","\rparsing log, completed traces ::  45%|####5     | 5903/13087 [00:04<00:04, 1718.92it/s]","\rparsing log, completed traces ::  46%|####6     | 6079/13087 [00:04<00:04, 1720.03it/s]","\rparsing log, completed traces ::  48%|####7     | 6257/13087 [00:04<00:03, 1734.49it/s]","\rparsing log, completed traces ::  49%|####9     | 6456/13087 [00:04<00:03, 1805.81it/s]","\rparsing log, completed traces ::  51%|#####     | 6639/13087 [00:04<00:03, 1734.63it/s]","\rparsing log, completed traces ::  52%|#####2    | 6815/13087 [00:04<00:03, 1721.32it/s]","\rparsing log, completed traces ::  53%|#####3    | 6989/13087 [00:04<00:03, 1642.69it/s]","\rparsing log, completed traces ::  55%|#####4    | 7155/13087 [00:05<00:07, 824.13it/s] ","\rparsing log, completed traces ::  56%|#####5    | 7315/13087 [00:05<00:06, 952.42it/s]","\rparsing log, completed traces ::  57%|#####6    | 7459/13087 [00:05<00:05, 1042.69it/s]","\rparsing log, completed traces ::  58%|#####8    | 7617/13087 [00:05<00:04, 1157.21it/s]","\rparsing log, completed traces ::  59%|#####9    | 7780/13087 [00:05<00:04, 1267.41it/s]","\rparsing log, completed traces ::  61%|######    | 7930/13087 [00:05<00:03, 1325.69it/s]","\rparsing log, completed traces ::  62%|######1   | 8104/13087 [00:05<00:03, 1431.71it/s]","\rparsing log, completed traces ::  63%|######3   | 8280/13087 [00:05<00:03, 1520.07it/s]","\rparsing log, completed traces ::  65%|######4   | 8457/13087 [00:05<00:02, 1587.74it/s]","\rparsing log, completed traces ::  66%|######5   | 8630/13087 [00:06<00:02, 1623.34it/s]","\rparsing log, completed traces ::  67%|######7   | 8801/13087 [00:06<00:02, 1645.92it/s]","\rparsing log, completed traces ::  69%|######8   | 8970/13087 [00:06<00:02, 1639.76it/s]","\rparsing log, completed traces ::  70%|######9   | 9137/13087 [00:06<00:02, 1604.00it/s]","\rparsing log, completed traces ::  71%|#######1  | 9331/13087 [00:06<00:02, 1699.16it/s]","\rparsing log, completed traces ::  73%|#######2  | 9503/13087 [00:06<00:02, 1672.33it/s]","\rparsing log, completed traces ::  74%|#######3  | 9672/13087 [00:06<00:02, 1627.88it/s]","\rparsing log, completed traces ::  75%|#######5  | 9852/13087 [00:06<00:01, 1674.76it/s]","\rparsing log, completed traces ::  77%|#######6  | 10021/13087 [00:06<00:01, 1642.00it/s]","\rparsing log, completed traces ::  78%|#######7  | 10186/13087 [00:07<00:01, 1624.42it/s]","\rparsing log, completed traces ::  79%|#######9  | 10349/13087 [00:07<00:01, 1593.10it/s]","\rparsing log, completed traces ::  80%|########  | 10509/13087 [00:07<00:01, 1558.55it/s]","\rparsing log, completed traces ::  82%|########1 | 10666/13087 [00:07<00:03, 721.11it/s] ","\rparsing log, completed traces ::  83%|########2 | 10840/13087 [00:07<00:02, 883.19it/s]","\rparsing log, completed traces ::  84%|########4 | 11031/13087 [00:07<00:01, 1072.90it/s]","\rparsing log, completed traces ::  86%|########5 | 11201/13087 [00:08<00:01, 1202.85it/s]","\rparsing log, completed traces ::  87%|########7 | 11387/13087 [00:08<00:01, 1353.31it/s]","\rparsing log, completed traces ::  88%|########8 | 11575/13087 [00:08<00:01, 1483.26it/s]","\rparsing log, completed traces ::  90%|########9 | 11766/13087 [00:08<00:00, 1593.85it/s]","\rparsing log, completed traces ::  91%|#########1| 11974/13087 [00:08<00:00, 1725.29it/s]","\rparsing log, completed traces ::  93%|#########2| 12161/13087 [00:08<00:00, 1763.37it/s]","\rparsing log, completed traces ::  94%|#########4| 12363/13087 [00:08<00:00, 1834.22it/s]","\rparsing log, completed traces ::  96%|#########6| 12565/13087 [00:08<00:00, 1886.97it/s]","\rparsing log, completed traces ::  98%|#########7| 12760/13087 [00:08<00:00, 1900.53it/s]","\rparsing log, completed traces ::  99%|#########8| 12955/13087 [00:08<00:00, 1908.73it/s]","","\rparsing log, completed traces :: 100%|##########| 13087/13087 [00:08<00:00, 1455.64it/s]","\n","[data] Loading XES: /workspace/data/BPI_Challenge_2017.xes","\n","\rparsing log, completed traces ::   0%|          | 0/31509 [00:00<?, ?it/s]","\rparsing log, completed traces ::   0%|          | 1/31509 [00:00<4:38:03,  1.89it/s]","\rparsing log, completed traces ::   0%|          | 68/31509 [00:00<03:39, 143.51it/s]","\rparsing log, completed traces ::   0%|          | 126/31509 [00:00<02:08, 244.61it/s]","\rparsing log, completed traces ::   1%|          | 194/31509 [00:00<01:28, 352.63it/s]","\rparsing log, completed traces ::   1%|          | 255/31509 [00:00<01:14, 420.04it/s]","\rparsing log, completed traces ::   1%|1         | 320/31509 [00:01<01:04, 480.14it/s]","\rparsing log, completed traces ::   1%|1         | 385/31509 [00:01<00:59, 527.38it/s]","\rparsing log, completed traces ::   1%|1         | 447/31509 [00:01<00:56, 546.25it/s]","\rparsing log, completed traces ::   2%|1         | 514/31509 [00:01<00:53, 580.35it/s]","\rparsing log, completed traces ::   2%|1         | 577/31509 [00:01<00:53, 579.52it/s]","\rparsing log, completed traces ::   2%|2         | 647/31509 [00:01<00:50, 611.56it/s]","\rparsing log, completed traces ::   2%|2         | 718/31509 [00:01<00:48, 639.34it/s]","\rparsing log, completed traces ::   2%|2         | 784/31509 [00:02<01:33, 328.87it/s]","\rparsing log, completed traces ::   3%|2         | 849/31509 [00:02<01:19, 384.97it/s]","\rparsing log, completed traces ::   3%|2         | 913/31509 [00:02<01:10, 435.49it/s]","\rparsing log, completed traces ::   3%|3         | 986/31509 [00:02<01:01, 498.59it/s]","\rparsing log, completed traces ::   3%|3         | 1048/31509 [00:02<00:57, 527.21it/s]","\rparsing log, completed traces ::   4%|3         | 1114/31509 [00:02<00:54, 559.39it/s]","\rparsing log, completed traces ::   4%|3         | 1184/31509 [00:02<00:50, 595.09it/s]","\rparsing log, completed traces ::   4%|3         | 1256/31509 [00:02<00:48, 627.34it/s]","\rparsing log, completed traces ::   4%|4         | 1323/31509 [00:02<00:48, 625.72it/s]","\rparsing log, completed traces ::   4%|4         | 1392/31509 [00:02<00:46, 643.64it/s]","\rparsing log, completed traces ::   5%|4         | 1459/31509 [00:03<00:47, 632.73it/s]","\rparsing log, completed traces ::   5%|4         | 1525/31509 [00:03<00:46, 640.42it/s]","\rparsing log, completed traces ::   5%|5         | 1591/31509 [00:03<00:46, 644.22it/s]","\rparsing log, completed traces ::   5%|5         | 1657/31509 [00:03<00:47, 631.51it/s]","\rparsing log, completed traces ::   5%|5         | 1723/31509 [00:03<00:46, 638.75it/s]","\rparsing log, completed traces ::   6%|5         | 1788/31509 [00:03<00:47, 627.45it/s]","\rparsing log, completed traces ::   6%|5         | 1852/31509 [00:03<00:48, 611.76it/s]","\rparsing log, completed traces ::   6%|6         | 1914/31509 [00:04<01:31, 324.78it/s]","\rparsing log, completed traces ::   6%|6         | 1981/31509 [00:04<01:16, 386.01it/s]","\rparsing log, completed traces ::   6%|6         | 2043/31509 [00:04<01:08, 432.90it/s]","\rparsing log, completed traces ::   7%|6         | 2107/31509 [00:04<01:01, 478.65it/s]","\rparsing log, completed traces ::   7%|6         | 2177/31509 [00:04<00:55, 530.91it/s]","\rparsing log, completed traces ::   7%|7         | 2245/31509 [00:04<00:51, 568.10it/s]","\rparsing log, completed traces ::   7%|7         | 2313/31509 [00:04<00:49, 593.85it/s]","\rparsing log, completed traces ::   8%|7         | 2378/31509 [00:04<00:48, 606.22it/s]","\rparsing log, completed traces ::   8%|7         | 2447/31509 [00:04<00:46, 628.10it/s]","\rparsing log, completed traces ::   8%|7         | 2513/31509 [00:05<00:45, 632.90it/s]","\rparsing log, completed traces ::   8%|8         | 2587/31509 [00:05<00:43, 661.22it/s]","\rparsing log, completed traces ::   8%|8         | 2657/31509 [00:05<00:42, 672.06it/s]","\rparsing log, completed traces ::   9%|8         | 2726/31509 [00:05<00:43, 663.86it/s]","\rparsing log, completed traces ::   9%|8         | 2797/31509 [00:05<00:42, 675.12it/s]","\rparsing log, completed traces ::   9%|9         | 2871/31509 [00:05<00:41, 693.25it/s]","\rparsing log, completed traces ::   9%|9         | 2943/31509 [00:05<00:40, 700.19it/s]","\rparsing log, completed traces ::  10%|9         | 3015/31509 [00:05<00:40, 704.83it/s]","\rparsing log, completed traces ::  10%|9         | 3086/31509 [00:05<00:41, 689.67it/s]","\rparsing log, completed traces ::  10%|#         | 3156/31509 [00:05<00:41, 680.49it/s]","\rparsing log, completed traces ::  10%|#         | 3225/31509 [00:06<00:42, 665.66it/s]","\rparsing log, completed traces ::  10%|#         | 3292/31509 [00:06<01:26, 325.63it/s]","\rparsing log, completed traces ::  11%|#         | 3365/31509 [00:06<01:11, 392.56it/s]","\rparsing log, completed traces ::  11%|#         | 3442/31509 [00:06<01:00, 465.25it/s]","\rparsing log, completed traces ::  11%|#1        | 3519/31509 [00:06<00:52, 530.10it/s]","\rparsing log, completed traces ::  11%|#1        | 3587/31509 [00:06<00:50, 556.31it/s]","\rparsing log, completed traces ::  12%|#1        | 3660/31509 [00:07<00:46, 597.35it/s]","\rparsing log, completed traces ::  12%|#1        | 3730/31509 [00:07<00:44, 623.32it/s]","\rparsing log, completed traces ::  12%|#2        | 3800/31509 [00:07<00:43, 642.26it/s]","\rparsing log, completed traces ::  12%|#2        | 3869/31509 [00:07<00:42, 647.28it/s]","\rparsing log, completed traces ::  13%|#2        | 3941/31509 [00:07<00:41, 666.39it/s]","\rparsing log, completed traces ::  13%|#2        | 4015/31509 [00:07<00:40, 685.62it/s]","\rparsing log, completed traces ::  13%|#2        | 4086/31509 [00:07<00:40, 683.75it/s]","\rparsing log, completed traces ::  13%|#3        | 4157/31509 [00:07<00:39, 690.57it/s]","\rparsing log, completed traces ::  13%|#3        | 4227/31509 [00:07<00:40, 670.02it/s]","\rparsing log, completed traces ::  14%|#3        | 4295/31509 [00:07<00:40, 665.47it/s]","\rparsing log, completed traces ::  14%|#3        | 4364/31509 [00:08<00:40, 671.91it/s]","\rparsing log, completed traces ::  14%|#4        | 4434/31509 [00:08<00:39, 678.47it/s]","\rparsing log, completed traces ::  14%|#4        | 4503/31509 [00:08<00:39, 678.79it/s]","\rparsing log, completed traces ::  15%|#4        | 4572/31509 [00:08<00:39, 676.40it/s]","\rparsing log, completed traces ::  15%|#4        | 4643/31509 [00:08<00:39, 685.89it/s]","\rparsing log, completed traces ::  15%|#4        | 4717/31509 [00:08<00:38, 697.92it/s]","\rparsing log, completed traces ::  15%|#5        | 4791/31509 [00:08<00:37, 708.20it/s]","\rparsing log, completed traces ::  15%|#5        | 4862/31509 [00:09<01:24, 317.17it/s]","\rparsing log, completed traces ::  16%|#5        | 4929/31509 [00:09<01:11, 372.00it/s]","\rparsing log, completed traces ::  16%|#5        | 4999/31509 [00:09<01:01, 432.00it/s]","\rparsing log, completed traces ::  16%|#6        | 5074/31509 [00:09<00:53, 497.05it/s]","\rparsing log, completed traces ::  16%|#6        | 5146/31509 [00:09<00:48, 546.37it/s]","\rparsing log, completed traces ::  17%|#6        | 5215/31509 [00:09<00:45, 579.83it/s]","\rparsing log, completed traces ::  17%|#6        | 5285/31509 [00:09<00:43, 609.73it/s]","\rparsing log, completed traces ::  17%|#6        | 5353/31509 [00:09<00:41, 625.56it/s]","\rparsing log, completed traces ::  17%|#7        | 5421/31509 [00:10<00:40, 637.32it/s]","\rparsing log, completed traces ::  17%|#7        | 5495/31509 [00:10<00:39, 665.14it/s]","\rparsing log, completed traces ::  18%|#7        | 5569/31509 [00:10<00:38, 681.68it/s]","\rparsing log, completed traces ::  18%|#7        | 5647/31509 [00:10<00:36, 709.27it/s]","\rparsing log, completed traces ::  18%|#8        | 5725/31509 [00:10<00:35, 729.66it/s]","\rparsing log, completed traces ::  18%|#8        | 5800/31509 [00:10<00:35, 716.62it/s]","\rparsing log, completed traces ::  19%|#8        | 5876/31509 [00:10<00:35, 728.72it/s]","\rparsing log, completed traces ::  19%|#8        | 5955/31509 [00:10<00:34, 743.20it/s]","\rparsing log, completed traces ::  19%|#9        | 6030/31509 [00:10<00:35, 721.54it/s]","\rparsing log, completed traces ::  19%|#9        | 6103/31509 [00:10<00:35, 710.51it/s]","\rparsing log, completed traces ::  20%|#9        | 6175/31509 [00:11<00:35, 708.48it/s]","\rparsing log, completed traces ::  20%|#9        | 6247/31509 [00:11<00:35, 708.54it/s]","\rparsing log, completed traces ::  20%|##        | 6318/31509 [00:11<00:36, 691.64it/s]","\rparsing log, completed traces ::  20%|##        | 6393/31509 [00:11<00:35, 705.56it/s]","\rparsing log, completed traces ::  21%|##        | 6464/31509 [00:11<00:35, 696.27it/s]","\rparsing log, completed traces ::  21%|##        | 6534/31509 [00:11<00:36, 675.45it/s]","\rparsing log, completed traces ::  21%|##        | 6602/31509 [00:12<01:24, 294.96it/s]","\rparsing log, completed traces ::  21%|##1       | 6672/31509 [00:12<01:09, 356.21it/s]","\rparsing log, completed traces ::  21%|##1       | 6744/31509 [00:12<00:58, 419.92it/s]","\rparsing log, completed traces ::  22%|##1       | 6811/31509 [00:12<00:52, 470.42it/s]","\rparsing log, completed traces ::  22%|##1       | 6879/31509 [00:12<00:47, 517.06it/s]","\rparsing log, completed traces ::  22%|##2       | 6950/31509 [00:12<00:44, 556.75it/s]","\rparsing log, completed traces ::  22%|##2       | 7026/31509 [00:12<00:40, 608.52it/s]","\rparsing log, completed traces ::  23%|##2       | 7096/31509 [00:12<00:38, 632.03it/s]","\rparsing log, completed traces ::  23%|##2       | 7166/31509 [00:12<00:37, 650.29it/s]","\rparsing log, completed traces ::  23%|##2       | 7236/31509 [00:13<00:37, 649.21it/s]","\rparsing log, completed traces ::  23%|##3       | 7307/31509 [00:13<00:36, 664.42it/s]","\rparsing log, completed traces ::  23%|##3       | 7376/31509 [00:13<00:36, 656.51it/s]","\rparsing log, completed traces ::  24%|##3       | 7445/31509 [00:13<00:36, 665.69it/s]","\rparsing log, completed traces ::  24%|##3       | 7521/31509 [00:13<00:34, 691.14it/s]","\rparsing log, completed traces ::  24%|##4       | 7595/31509 [00:13<00:33, 704.38it/s]","\rparsing log, completed traces ::  24%|##4       | 7667/31509 [00:13<00:35, 678.10it/s]","\rparsing log, completed traces ::  25%|##4       | 7736/31509 [00:13<00:35, 671.17it/s]","\rparsing log, completed traces ::  25%|##4       | 7804/31509 [00:13<00:36, 656.35it/s]","\rparsing log, completed traces ::  25%|##5       | 7878/31509 [00:14<00:34, 678.67it/s]","\rparsing log, completed traces ::  25%|##5       | 7947/31509 [00:14<00:34, 679.65it/s]","\rparsing log, completed traces ::  25%|##5       | 8019/31509 [00:14<00:34, 689.74it/s]","\rparsing log, completed traces ::  26%|##5       | 8091/31509 [00:14<00:33, 693.75it/s]","\rparsing log, completed traces ::  26%|##5       | 8161/31509 [00:14<00:33, 693.17it/s]","\rparsing log, completed traces ::  26%|##6       | 8236/31509 [00:14<00:32, 708.03it/s]","\rparsing log, completed traces ::  26%|##6       | 8307/31509 [00:14<00:32, 705.85it/s]","\rparsing log, completed traces ::  27%|##6       | 8378/31509 [00:14<00:33, 692.92it/s]","\rparsing log, completed traces ::  27%|##6       | 8448/31509 [00:14<00:33, 687.19it/s]","\rparsing log, completed traces ::  27%|##7       | 8517/31509 [00:15<01:24, 270.89it/s]","\rparsing log, completed traces ::  27%|##7       | 8581/31509 [00:15<01:11, 322.78it/s]","\rparsing log, completed traces ::  27%|##7       | 8649/31509 [00:15<00:59, 382.08it/s]","\rparsing log, completed traces ::  28%|##7       | 8719/31509 [00:15<00:51, 443.48it/s]","\rparsing log, completed traces ::  28%|##7       | 8790/31509 [00:15<00:45, 499.38it/s]","\rparsing log, completed traces ::  28%|##8       | 8858/31509 [00:15<00:41, 540.55it/s]","\rparsing log, completed traces ::  28%|##8       | 8924/31509 [00:16<00:39, 567.41it/s]","\rparsing log, completed traces ::  29%|##8       | 8993/31509 [00:16<00:37, 596.51it/s]","\rparsing log, completed traces ::  29%|##8       | 9059/31509 [00:16<00:37, 594.31it/s]","\rparsing log, completed traces ::  29%|##8       | 9127/31509 [00:16<00:36, 617.47it/s]","\rparsing log, completed traces ::  29%|##9       | 9195/31509 [00:16<00:35, 634.50it/s]","\rparsing log, completed traces ::  29%|##9       | 9261/31509 [00:16<00:35, 621.88it/s]","\rparsing log, completed traces ::  30%|##9       | 9335/31509 [00:16<00:33, 655.02it/s]","\rparsing log, completed traces ::  30%|##9       | 9402/31509 [00:16<00:34, 641.73it/s]","\rparsing log, completed traces ::  30%|###       | 9471/31509 [00:16<00:33, 654.78it/s]","\rparsing log, completed traces ::  30%|###       | 9538/31509 [00:16<00:33, 658.07it/s]","\rparsing log, completed traces ::  30%|###       | 9605/31509 [00:17<00:33, 649.38it/s]","\rparsing log, completed traces ::  31%|###       | 9673/31509 [00:17<00:33, 656.86it/s]","\rparsing log, completed traces ::  31%|###       | 9739/31509 [00:17<00:33, 643.70it/s]","\rparsing log, completed traces ::  31%|###1      | 9805/31509 [00:17<00:33, 648.16it/s]","\rparsing log, completed traces ::  31%|###1      | 9870/31509 [00:17<00:33, 646.52it/s]","\rparsing log, completed traces ::  32%|###1      | 9935/31509 [00:17<00:33, 643.05it/s]","\rparsing log, completed traces ::  32%|###1      | 10000/31509 [00:17<00:33, 635.92it/s]","\rparsing log, completed traces ::  32%|###1      | 10064/31509 [00:17<00:33, 635.26it/s]","\rparsing log, completed traces ::  32%|###2      | 10128/31509 [00:17<00:34, 620.00it/s]","\rparsing log, completed traces ::  32%|###2      | 10196/31509 [00:18<00:33, 636.25it/s]","\rparsing log, completed traces ::  33%|###2      | 10266/31509 [00:18<00:32, 653.04it/s]","\rparsing log, completed traces ::  33%|###2      | 10332/31509 [00:18<00:33, 629.97it/s]","\rparsing log, completed traces ::  33%|###2      | 10396/31509 [00:18<00:33, 625.83it/s]","\rparsing log, completed traces ::  33%|###3      | 10460/31509 [00:18<00:33, 627.65it/s]","\rparsing log, completed traces ::  33%|###3      | 10523/31509 [00:19<01:29, 234.62it/s]","\rparsing log, completed traces ::  34%|###3      | 10590/31509 [00:19<01:11, 293.12it/s]","\rparsing log, completed traces ::  34%|###3      | 10649/31509 [00:19<01:01, 340.91it/s]","\rparsing log, completed traces ::  34%|###3      | 10713/31509 [00:19<00:52, 396.68it/s]","\rparsing log, completed traces ::  34%|###4      | 10775/31509 [00:19<00:46, 443.11it/s]","\rparsing log, completed traces ::  34%|###4      | 10834/31509 [00:19<00:43, 473.74it/s]","\rparsing log, completed traces ::  35%|###4      | 10893/31509 [00:19<00:41, 500.86it/s]","\rparsing log, completed traces ::  35%|###4      | 10955/31509 [00:19<00:38, 528.16it/s]","\rparsing log, completed traces ::  35%|###4      | 11025/31509 [00:19<00:35, 573.49it/s]","\rparsing log, completed traces ::  35%|###5      | 11088/31509 [00:20<00:34, 587.80it/s]","\rparsing log, completed traces ::  35%|###5      | 11151/31509 [00:20<00:34, 596.84it/s]","\rparsing log, completed traces ::  36%|###5      | 11214/31509 [00:20<00:33, 604.10it/s]","\rparsing log, completed traces ::  36%|###5      | 11282/31509 [00:20<00:32, 624.25it/s]","\rparsing log, completed traces ::  36%|###6      | 11346/31509 [00:20<00:32, 627.94it/s]","\rparsing log, completed traces ::  36%|###6      | 11410/31509 [00:20<00:32, 620.73it/s]","\rparsing log, completed traces ::  36%|###6      | 11473/31509 [00:20<00:32, 616.68it/s]","\rparsing log, completed traces ::  37%|###6      | 11536/31509 [00:20<00:33, 603.09it/s]","\rparsing log, completed traces ::  37%|###6      | 11600/31509 [00:20<00:32, 612.09it/s]","\rparsing log, completed traces ::  37%|###7      | 11662/31509 [00:20<00:32, 609.66it/s]","\rparsing log, completed traces ::  37%|###7      | 11724/31509 [00:21<00:33, 595.03it/s]","\rparsing log, completed traces ::  37%|###7      | 11786/31509 [00:21<00:32, 601.39it/s]","\rparsing log, completed traces ::  38%|###7      | 11847/31509 [00:21<00:32, 599.19it/s]","\rparsing log, completed traces ::  38%|###7      | 11914/31509 [00:21<00:31, 619.26it/s]","\rparsing log, completed traces ::  38%|###8      | 11978/31509 [00:21<00:31, 625.30it/s]","\rparsing log, completed traces ::  38%|###8      | 12044/31509 [00:21<00:30, 634.34it/s]","\rparsing log, completed traces ::  38%|###8      | 12108/31509 [00:21<00:30, 632.39it/s]","\rparsing log, completed traces ::  39%|###8      | 12173/31509 [00:21<00:30, 634.04it/s]","\rparsing log, completed traces ::  39%|###8      | 12245/31509 [00:21<00:29, 657.98it/s]","\rparsing log, completed traces ::  39%|###9      | 12312/31509 [00:21<00:29, 661.36it/s]","\rparsing log, completed traces ::  39%|###9      | 12380/31509 [00:22<00:28, 666.21it/s]","\rparsing log, completed traces ::  40%|###9      | 12448/31509 [00:22<00:28, 668.67it/s]","\rparsing log, completed traces ::  40%|###9      | 12515/31509 [00:22<00:28, 657.51it/s]","\rparsing log, completed traces ::  40%|###9      | 12582/31509 [00:22<00:28, 659.69it/s]","\rparsing log, completed traces ::  40%|####      | 12649/31509 [00:22<00:28, 660.27it/s]","\rparsing log, completed traces ::  40%|####      | 12720/31509 [00:22<00:27, 674.68it/s]","\rparsing log, completed traces ::  41%|####      | 12788/31509 [00:22<00:27, 669.40it/s]","\rparsing log, completed traces ::  41%|####      | 12855/31509 [00:23<01:17, 239.77it/s]","\rparsing log, completed traces ::  41%|####      | 12918/31509 [00:23<01:03, 290.89it/s]","\rparsing log, completed traces ::  41%|####1     | 12990/31509 [00:23<00:51, 358.11it/s]","\rparsing log, completed traces ::  41%|####1     | 13059/31509 [00:23<00:44, 418.32it/s]","\rparsing log, completed traces ::  42%|####1     | 13130/31509 [00:23<00:38, 478.88it/s]","\rparsing log, completed traces ::  42%|####1     | 13195/31509 [00:23<00:35, 514.98it/s]","\rparsing log, completed traces ::  42%|####2     | 13260/31509 [00:23<00:33, 547.91it/s]","\rparsing log, completed traces ::  42%|####2     | 13329/31509 [00:24<00:31, 582.56it/s]","\rparsing log, completed traces ::  43%|####2     | 13398/31509 [00:24<00:29, 611.33it/s]","\rparsing log, completed traces ::  43%|####2     | 13465/31509 [00:24<00:29, 617.70it/s]","\rparsing log, completed traces ::  43%|####2     | 13531/31509 [00:24<00:28, 623.68it/s]","\rparsing log, completed traces ::  43%|####3     | 13597/31509 [00:24<00:28, 625.55it/s]","\rparsing log, completed traces ::  43%|####3     | 13662/31509 [00:24<00:28, 626.68it/s]","\rparsing log, completed traces ::  44%|####3     | 13728/31509 [00:24<00:28, 634.34it/s]","\rparsing log, completed traces ::  44%|####3     | 13794/31509 [00:24<00:27, 641.26it/s]","\rparsing log, completed traces ::  44%|####3     | 13861/31509 [00:24<00:27, 647.49it/s]","\rparsing log, completed traces ::  44%|####4     | 13927/31509 [00:25<00:27, 645.84it/s]","\rparsing log, completed traces ::  44%|####4     | 13994/31509 [00:25<00:26, 652.61it/s]","\rparsing log, completed traces ::  45%|####4     | 14065/31509 [00:25<00:26, 669.06it/s]","\rparsing log, completed traces ::  45%|####4     | 14133/31509 [00:25<00:26, 660.17it/s]","\rparsing log, completed traces ::  45%|####5     | 14200/31509 [00:25<00:26, 657.12it/s]","\rparsing log, completed traces ::  45%|####5     | 14266/31509 [00:25<00:26, 654.27it/s]","\rparsing log, completed traces ::  45%|####5     | 14332/31509 [00:25<00:26, 653.39it/s]","\rparsing log, completed traces ::  46%|####5     | 14405/31509 [00:25<00:25, 674.85it/s]","\rparsing log, completed traces ::  46%|####5     | 14473/31509 [00:25<00:25, 664.85it/s]","\rparsing log, completed traces ::  46%|####6     | 14540/31509 [00:25<00:26, 645.50it/s]","\rparsing log, completed traces ::  46%|####6     | 14605/31509 [00:26<00:26, 638.90it/s]","\rparsing log, completed traces ::  47%|####6     | 14670/31509 [00:26<00:26, 639.52it/s]","\rparsing log, completed traces ::  47%|####6     | 14739/31509 [00:26<00:25, 649.59it/s]","\rparsing log, completed traces ::  47%|####7     | 14810/31509 [00:26<00:25, 665.03it/s]","\rparsing log, completed traces ::  47%|####7     | 14877/31509 [00:26<00:25, 649.97it/s]","\rparsing log, completed traces ::  47%|####7     | 14953/31509 [00:26<00:24, 679.36it/s]","\rparsing log, completed traces ::  48%|####7     | 15022/31509 [00:26<00:25, 657.48it/s]","\rparsing log, completed traces ::  48%|####7     | 15096/31509 [00:26<00:24, 680.12it/s]","\rparsing log, completed traces ::  48%|####8     | 15165/31509 [00:26<00:23, 681.90it/s]","\rparsing log, completed traces ::  48%|####8     | 15234/31509 [00:26<00:23, 680.25it/s]","\rparsing log, completed traces ::  49%|####8     | 15309/31509 [00:27<00:23, 700.09it/s]","\rparsing log, completed traces ::  49%|####8     | 15382/31509 [00:27<00:22, 707.42it/s]","\rparsing log, completed traces ::  49%|####9     | 15455/31509 [00:27<00:22, 712.38it/s]","\rparsing log, completed traces ::  49%|####9     | 15527/31509 [00:28<01:08, 234.63it/s]","\rparsing log, completed traces ::  50%|####9     | 15600/31509 [00:28<00:53, 294.73it/s]","\rparsing log, completed traces ::  50%|####9     | 15676/31509 [00:28<00:43, 362.55it/s]","\rparsing log, completed traces ::  50%|####9     | 15744/31509 [00:28<00:37, 417.38it/s]","\rparsing log, completed traces ::  50%|#####     | 15815/31509 [00:28<00:33, 474.48it/s]","\rparsing log, completed traces ::  50%|#####     | 15889/31509 [00:28<00:29, 532.34it/s]","\rparsing log, completed traces ::  51%|#####     | 15959/31509 [00:28<00:27, 572.44it/s]","\rparsing log, completed traces ::  51%|#####     | 16028/31509 [00:28<00:25, 597.01it/s]","\rparsing log, completed traces ::  51%|#####1    | 16103/31509 [00:28<00:24, 636.71it/s]","\rparsing log, completed traces ::  51%|#####1    | 16180/31509 [00:28<00:22, 672.28it/s]","\rparsing log, completed traces ::  52%|#####1    | 16253/31509 [00:29<00:22, 681.72it/s]","\rparsing log, completed traces ::  52%|#####1    | 16327/31509 [00:29<00:21, 696.25it/s]","\rparsing log, completed traces ::  52%|#####2    | 16400/31509 [00:29<00:22, 670.84it/s]","\rparsing log, completed traces ::  52%|#####2    | 16470/31509 [00:29<00:22, 677.51it/s]","\rparsing log, completed traces ::  52%|#####2    | 16540/31509 [00:29<00:22, 663.35it/s]","\rparsing log, completed traces ::  53%|#####2    | 16611/31509 [00:29<00:22, 674.37it/s]","\rparsing log, completed traces ::  53%|#####2    | 16685/31509 [00:29<00:21, 692.26it/s]","\rparsing log, completed traces ::  53%|#####3    | 16755/31509 [00:29<00:21, 684.77it/s]","\rparsing log, completed traces ::  53%|#####3    | 16826/31509 [00:29<00:21, 690.45it/s]","\rparsing log, completed traces ::  54%|#####3    | 16898/31509 [00:30<00:20, 699.10it/s]","\rparsing log, completed traces ::  54%|#####3    | 16969/31509 [00:30<00:20, 699.42it/s]","\rparsing log, completed traces ::  54%|#####4    | 17045/31509 [00:30<00:20, 717.26it/s]","\rparsing log, completed traces ::  54%|#####4    | 17117/31509 [00:30<00:20, 712.98it/s]","\rparsing log, completed traces ::  55%|#####4    | 17189/31509 [00:30<00:20, 704.26it/s]","\rparsing log, completed traces ::  55%|#####4    | 17260/31509 [00:30<00:20, 698.23it/s]","\rparsing log, completed traces ::  55%|#####5    | 17332/31509 [00:30<00:20, 703.73it/s]","\rparsing log, completed traces ::  55%|#####5    | 17407/31509 [00:30<00:19, 716.78it/s]","\rparsing log, completed traces ::  55%|#####5    | 17479/31509 [00:30<00:19, 706.98it/s]","\rparsing log, completed traces ::  56%|#####5    | 17550/31509 [00:30<00:20, 695.96it/s]","\rparsing log, completed traces ::  56%|#####5    | 17629/31509 [00:31<00:19, 723.06it/s]","\rparsing log, completed traces ::  56%|#####6    | 17702/31509 [00:31<00:19, 721.77it/s]","\rparsing log, completed traces ::  56%|#####6    | 17775/31509 [00:31<00:19, 719.16it/s]","\rparsing log, completed traces ::  57%|#####6    | 17851/31509 [00:31<00:18, 730.92it/s]","\rparsing log, completed traces ::  57%|#####6    | 17925/31509 [00:31<00:18, 719.67it/s]","\rparsing log, completed traces ::  57%|#####7    | 18001/31509 [00:31<00:18, 728.43it/s]","\rparsing log, completed traces ::  57%|#####7    | 18077/31509 [00:31<00:18, 736.66it/s]","\rparsing log, completed traces ::  58%|#####7    | 18151/31509 [00:31<00:18, 731.31it/s]","\rparsing log, completed traces ::  58%|#####7    | 18225/31509 [00:31<00:18, 721.18it/s]","\rparsing log, completed traces ::  58%|#####8    | 18298/31509 [00:31<00:18, 722.47it/s]","\rparsing log, completed traces ::  58%|#####8    | 18371/31509 [00:32<00:18, 708.93it/s]","\rparsing log, completed traces ::  59%|#####8    | 18445/31509 [00:32<00:18, 717.62it/s]","\rparsing log, completed traces ::  59%|#####8    | 18517/31509 [00:32<00:18, 702.50it/s]","\rparsing log, completed traces ::  59%|#####8    | 18588/31509 [00:32<00:18, 688.40it/s]","\rparsing log, completed traces ::  59%|#####9    | 18662/31509 [00:33<00:59, 215.82it/s]","\rparsing log, completed traces ::  59%|#####9    | 18739/31509 [00:33<00:46, 276.91it/s]","\rparsing log, completed traces ::  60%|#####9    | 18808/31509 [00:33<00:38, 332.67it/s]","\rparsing log, completed traces ::  60%|#####9    | 18884/31509 [00:33<00:31, 402.36it/s]","\rparsing log, completed traces ::  60%|######    | 18959/31509 [00:33<00:26, 467.18it/s]","\rparsing log, completed traces ::  60%|######    | 19032/31509 [00:33<00:23, 522.17it/s]","\rparsing log, completed traces ::  61%|######    | 19106/31509 [00:33<00:21, 571.50it/s]","\rparsing log, completed traces ::  61%|######    | 19179/31509 [00:33<00:20, 607.37it/s]","\rparsing log, completed traces ::  61%|######1   | 19250/31509 [00:34<00:19, 631.29it/s]","\rparsing log, completed traces ::  61%|######1   | 19321/31509 [00:34<00:18, 651.71it/s]","\rparsing log, completed traces ::  62%|######1   | 19398/31509 [00:34<00:17, 684.20it/s]","\rparsing log, completed traces ::  62%|######1   | 19471/31509 [00:34<00:17, 691.11it/s]","\rparsing log, completed traces ::  62%|######2   | 19544/31509 [00:34<00:17, 699.99it/s]","\rparsing log, completed traces ::  62%|######2   | 19616/31509 [00:34<00:16, 703.61it/s]","\rparsing log, completed traces ::  62%|######2   | 19688/31509 [00:34<00:17, 690.17it/s]","\rparsing log, completed traces ::  63%|######2   | 19759/31509 [00:34<00:16, 694.78it/s]","\rparsing log, completed traces ::  63%|######2   | 19830/31509 [00:34<00:16, 694.02it/s]","\rparsing log, completed traces ::  63%|######3   | 19900/31509 [00:35<00:17, 678.93it/s]","\rparsing log, completed traces ::  63%|######3   | 19970/31509 [00:35<00:16, 683.80it/s]","\rparsing log, completed traces ::  64%|######3   | 20040/31509 [00:35<00:16, 687.39it/s]","\rparsing log, completed traces ::  64%|######3   | 20109/31509 [00:35<00:16, 679.17it/s]","\rparsing log, completed traces ::  64%|######4   | 20181/31509 [00:35<00:16, 689.88it/s]","\rparsing log, completed traces ::  64%|######4   | 20254/31509 [00:35<00:16, 701.34it/s]","\rparsing log, completed traces ::  65%|######4   | 20325/31509 [00:35<00:16, 698.15it/s]","\rparsing log, completed traces ::  65%|######4   | 20401/31509 [00:35<00:15, 716.07it/s]","\rparsing log, completed traces ::  65%|######4   | 20476/31509 [00:35<00:15, 722.30it/s]","\rparsing log, completed traces ::  65%|######5   | 20549/31509 [00:35<00:15, 718.58it/s]","\rparsing log, completed traces ::  65%|######5   | 20627/31509 [00:36<00:14, 736.24it/s]","\rparsing log, completed traces ::  66%|######5   | 20705/31509 [00:36<00:14, 747.19it/s]","\rparsing log, completed traces ::  66%|######5   | 20780/31509 [00:36<00:14, 739.15it/s]","\rparsing log, completed traces ::  66%|######6   | 20854/31509 [00:36<00:14, 726.90it/s]","\rparsing log, completed traces ::  66%|######6   | 20927/31509 [00:36<00:14, 714.40it/s]","\rparsing log, completed traces ::  67%|######6   | 20999/31509 [00:36<00:15, 697.14it/s]","\rparsing log, completed traces ::  67%|######6   | 21069/31509 [00:36<00:15, 690.75it/s]","\rparsing log, completed traces ::  67%|######7   | 21139/31509 [00:36<00:15, 686.19it/s]","\rparsing log, completed traces ::  67%|######7   | 21212/31509 [00:36<00:14, 696.44it/s]","\rparsing log, completed traces ::  68%|######7   | 21286/31509 [00:36<00:14, 706.68it/s]","\rparsing log, completed traces ::  68%|######7   | 21357/31509 [00:37<00:14, 704.67it/s]","\rparsing log, completed traces ::  68%|######8   | 21429/31509 [00:37<00:14, 708.38it/s]","\rparsing log, completed traces ::  68%|######8   | 21507/31509 [00:37<00:13, 728.88it/s]","\rparsing log, completed traces ::  68%|######8   | 21583/31509 [00:37<00:13, 737.65it/s]","\rparsing log, completed traces ::  69%|######8   | 21657/31509 [00:37<00:13, 716.81it/s]","\rparsing log, completed traces ::  69%|######8   | 21733/31509 [00:37<00:13, 725.97it/s]","\rparsing log, completed traces ::  69%|######9   | 21807/31509 [00:37<00:13, 724.72it/s]","\rparsing log, completed traces ::  69%|######9   | 21889/31509 [00:37<00:12, 750.43it/s]","\rparsing log, completed traces ::  70%|######9   | 21965/31509 [00:37<00:12, 734.27it/s]","\rparsing log, completed traces ::  70%|######9   | 22046/31509 [00:38<00:12, 753.34it/s]","\rparsing log, completed traces ::  70%|#######   | 22122/31509 [00:38<00:12, 746.10it/s]","\rparsing log, completed traces ::  70%|#######   | 22197/31509 [00:38<00:12, 746.93it/s]","\rparsing log, completed traces ::  71%|#######   | 22272/31509 [00:39<00:44, 207.97it/s]","\rparsing log, completed traces ::  71%|#######   | 22345/31509 [00:39<00:34, 262.59it/s]","\rparsing log, completed traces ::  71%|#######1  | 22422/31509 [00:39<00:27, 328.29it/s]","\rparsing log, completed traces ::  71%|#######1  | 22498/31509 [00:39<00:22, 395.02it/s]","\rparsing log, completed traces ::  72%|#######1  | 22569/31509 [00:39<00:19, 452.39it/s]","\rparsing log, completed traces ::  72%|#######1  | 22643/31509 [00:39<00:17, 509.28it/s]","\rparsing log, completed traces ::  72%|#######2  | 22716/31509 [00:39<00:15, 557.17it/s]","\rparsing log, completed traces ::  72%|#######2  | 22792/31509 [00:39<00:14, 605.92it/s]","\rparsing log, completed traces ::  73%|#######2  | 22867/31509 [00:39<00:13, 641.19it/s]","\rparsing log, completed traces ::  73%|#######2  | 22940/31509 [00:40<00:13, 640.25it/s]","\rparsing log, completed traces ::  73%|#######3  | 23010/31509 [00:40<00:12, 653.84it/s]","\rparsing log, completed traces ::  73%|#######3  | 23082/31509 [00:40<00:12, 672.13it/s]","\rparsing log, completed traces ::  73%|#######3  | 23153/31509 [00:40<00:12, 675.23it/s]","\rparsing log, completed traces ::  74%|#######3  | 23223/31509 [00:40<00:12, 679.63it/s]","\rparsing log, completed traces ::  74%|#######3  | 23293/31509 [00:40<00:12, 677.63it/s]","\rparsing log, completed traces ::  74%|#######4  | 23362/31509 [00:40<00:12, 666.51it/s]","\rparsing log, completed traces ::  74%|#######4  | 23434/31509 [00:40<00:11, 679.30it/s]","\rparsing log, completed traces ::  75%|#######4  | 23503/31509 [00:40<00:11, 680.47it/s]","\rparsing log, completed traces ::  75%|#######4  | 23577/31509 [00:41<00:11, 697.41it/s]","\rparsing log, completed traces ::  75%|#######5  | 23649/31509 [00:41<00:11, 702.07it/s]","\rparsing log, completed traces ::  75%|#######5  | 23720/31509 [00:41<00:11, 700.65it/s]","\rparsing log, completed traces ::  76%|#######5  | 23791/31509 [00:41<00:10, 702.44it/s]","\rparsing log, completed traces ::  76%|#######5  | 23862/31509 [00:41<00:10, 700.77it/s]","\rparsing log, completed traces ::  76%|#######5  | 23933/31509 [00:41<00:10, 695.08it/s]","\rparsing log, completed traces ::  76%|#######6  | 24003/31509 [00:41<00:10, 695.16it/s]","\rparsing log, completed traces ::  76%|#######6  | 24074/31509 [00:41<00:10, 695.59it/s]","\rparsing log, completed traces ::  77%|#######6  | 24144/31509 [00:41<00:10, 695.93it/s]","\rparsing log, completed traces ::  77%|#######6  | 24215/31509 [00:41<00:10, 699.66it/s]","\rparsing log, completed traces ::  77%|#######7  | 24285/31509 [00:42<00:10, 699.08it/s]","\rparsing log, completed traces ::  77%|#######7  | 24360/31509 [00:42<00:10, 712.22it/s]","\rparsing log, completed traces ::  78%|#######7  | 24432/31509 [00:42<00:10, 692.82it/s]","\rparsing log, completed traces ::  78%|#######7  | 24507/31509 [00:42<00:09, 708.25it/s]","\rparsing log, completed traces ::  78%|#######8  | 24581/31509 [00:42<00:09, 715.18it/s]","\rparsing log, completed traces ::  78%|#######8  | 24658/31509 [00:42<00:09, 726.88it/s]","\rparsing log, completed traces ::  78%|#######8  | 24731/31509 [00:42<00:09, 717.91it/s]","\rparsing log, completed traces ::  79%|#######8  | 24803/31509 [00:42<00:09, 718.46it/s]","\rparsing log, completed traces ::  79%|#######8  | 24881/31509 [00:42<00:09, 734.72it/s]","\rparsing log, completed traces ::  79%|#######9  | 24955/31509 [00:42<00:09, 725.90it/s]","\rparsing log, completed traces ::  79%|#######9  | 25032/31509 [00:43<00:08, 737.84it/s]","\rparsing log, completed traces ::  80%|#######9  | 25106/31509 [00:43<00:09, 707.23it/s]","\rparsing log, completed traces ::  80%|#######9  | 25183/31509 [00:43<00:08, 724.57it/s]","\rparsing log, completed traces ::  80%|########  | 25256/31509 [00:43<00:08, 710.30it/s]","\rparsing log, completed traces ::  80%|########  | 25328/31509 [00:43<00:09, 686.26it/s]","\rparsing log, completed traces ::  81%|########  | 25405/31509 [00:43<00:08, 708.68it/s]","\rparsing log, completed traces ::  81%|########  | 25479/31509 [00:43<00:08, 714.16it/s]","\rparsing log, completed traces ::  81%|########1 | 25551/31509 [00:43<00:08, 712.03it/s]","\rparsing log, completed traces ::  81%|########1 | 25623/31509 [00:43<00:08, 702.72it/s]","\rparsing log, completed traces ::  82%|########1 | 25694/31509 [00:44<00:08, 701.70it/s]","\rparsing log, completed traces ::  82%|########1 | 25766/31509 [00:44<00:08, 706.83it/s]","\rparsing log, completed traces ::  82%|########1 | 25837/31509 [00:44<00:08, 689.44it/s]","\rparsing log, completed traces ::  82%|########2 | 25908/31509 [00:44<00:08, 693.81it/s]","\rparsing log, completed traces ::  82%|########2 | 25978/31509 [00:44<00:07, 691.63it/s]","\rparsing log, completed traces ::  83%|########2 | 26048/31509 [00:44<00:07, 684.05it/s]","\rparsing log, completed traces ::  83%|########2 | 26117/31509 [00:44<00:07, 678.18it/s]","\rparsing log, completed traces ::  83%|########3 | 26189/31509 [00:44<00:07, 688.99it/s]","\rparsing log, completed traces ::  83%|########3 | 26258/31509 [00:45<00:29, 178.60it/s]","\rparsing log, completed traces ::  84%|########3 | 26331/31509 [00:45<00:22, 232.48it/s]","\rparsing log, completed traces ::  84%|########3 | 26404/31509 [00:46<00:17, 293.53it/s]","\rparsing log, completed traces ::  84%|########4 | 26472/31509 [00:46<00:14, 351.02it/s]","\rparsing log, completed traces ::  84%|########4 | 26536/31509 [00:46<00:12, 401.40it/s]","\rparsing log, completed traces ::  84%|########4 | 26602/31509 [00:46<00:10, 452.01it/s]","\rparsing log, completed traces ::  85%|########4 | 26673/31509 [00:46<00:09, 508.41it/s]","\rparsing log, completed traces ::  85%|########4 | 26746/31509 [00:46<00:08, 560.57it/s]","\rparsing log, completed traces ::  85%|########5 | 26814/31509 [00:46<00:07, 587.00it/s]","\rparsing log, completed traces ::  85%|########5 | 26882/31509 [00:46<00:07, 611.01it/s]","\rparsing log, completed traces ::  86%|########5 | 26953/31509 [00:46<00:07, 634.57it/s]","\rparsing log, completed traces ::  86%|########5 | 27022/31509 [00:46<00:07, 636.54it/s]","\rparsing log, completed traces ::  86%|########5 | 27089/31509 [00:47<00:06, 645.10it/s]","\rparsing log, completed traces ::  86%|########6 | 27164/31509 [00:47<00:06, 674.63it/s]","\rparsing log, completed traces ::  86%|########6 | 27236/31509 [00:47<00:06, 686.74it/s]","\rparsing log, completed traces ::  87%|########6 | 27306/31509 [00:47<00:06, 683.46it/s]","\rparsing log, completed traces ::  87%|########6 | 27376/31509 [00:47<00:06, 683.27it/s]","\rparsing log, completed traces ::  87%|########7 | 27446/31509 [00:47<00:05, 684.85it/s]","\rparsing log, completed traces ::  87%|########7 | 27515/31509 [00:47<00:05, 683.60it/s]","\rparsing log, completed traces ::  88%|########7 | 27590/31509 [00:47<00:05, 701.59it/s]","\rparsing log, completed traces ::  88%|########7 | 27663/31509 [00:47<00:05, 707.37it/s]","\rparsing log, completed traces ::  88%|########8 | 27740/31509 [00:47<00:05, 723.31it/s]","\rparsing log, completed traces ::  88%|########8 | 27813/31509 [00:48<00:05, 708.30it/s]","\rparsing log, completed traces ::  88%|########8 | 27884/31509 [00:48<00:05, 698.05it/s]","\rparsing log, completed traces ::  89%|########8 | 27954/31509 [00:48<00:05, 697.45it/s]","\rparsing log, completed traces ::  89%|########8 | 28024/31509 [00:48<00:05, 696.17it/s]","\rparsing log, completed traces ::  89%|########9 | 28094/31509 [00:48<00:05, 681.64it/s]","\rparsing log, completed traces ::  89%|########9 | 28163/31509 [00:48<00:04, 670.99it/s]","\rparsing log, completed traces ::  90%|########9 | 28233/31509 [00:48<00:04, 678.63it/s]","\rparsing log, completed traces ::  90%|########9 | 28301/31509 [00:48<00:04, 674.09it/s]","\rparsing log, completed traces ::  90%|######### | 28372/31509 [00:48<00:04, 682.57it/s]","\rparsing log, completed traces ::  90%|######### | 28441/31509 [00:48<00:04, 673.90it/s]","\rparsing log, completed traces ::  91%|######### | 28516/31509 [00:49<00:04, 695.51it/s]","\rparsing log, completed traces ::  91%|######### | 28591/31509 [00:49<00:04, 706.07it/s]","\rparsing log, completed traces ::  91%|######### | 28662/31509 [00:49<00:04, 698.01it/s]","\rparsing log, completed traces ::  91%|#########1| 28732/31509 [00:49<00:04, 673.80it/s]","\rparsing log, completed traces ::  91%|#########1| 28800/31509 [00:49<00:04, 658.88it/s]","\rparsing log, completed traces ::  92%|#########1| 28867/31509 [00:49<00:04, 644.49it/s]","\rparsing log, completed traces ::  92%|#########1| 28937/31509 [00:49<00:03, 659.56it/s]","\rparsing log, completed traces ::  92%|#########2| 29008/31509 [00:49<00:03, 673.42it/s]","\rparsing log, completed traces ::  92%|#########2| 29076/31509 [00:49<00:03, 658.90it/s]","\rparsing log, completed traces ::  92%|#########2| 29143/31509 [00:50<00:03, 659.99it/s]","\rparsing log, completed traces ::  93%|#########2| 29219/31509 [00:50<00:03, 685.84it/s]","\rparsing log, completed traces ::  93%|#########2| 29288/31509 [00:50<00:03, 678.65it/s]","\rparsing log, completed traces ::  93%|#########3| 29356/31509 [00:50<00:03, 678.91it/s]","\rparsing log, completed traces ::  93%|#########3| 29424/31509 [00:50<00:03, 661.62it/s]","\rparsing log, completed traces ::  94%|#########3| 29491/31509 [00:50<00:03, 651.41it/s]","\rparsing log, completed traces ::  94%|#########3| 29557/31509 [00:50<00:03, 629.61it/s]","\rparsing log, completed traces ::  94%|#########4| 29633/31509 [00:50<00:02, 666.20it/s]","\rparsing log, completed traces ::  94%|#########4| 29703/31509 [00:50<00:02, 675.40it/s]","\rparsing log, completed traces ::  94%|#########4| 29771/31509 [00:50<00:02, 673.51it/s]","\rparsing log, completed traces ::  95%|#########4| 29839/31509 [00:51<00:02, 667.05it/s]","\rparsing log, completed traces ::  95%|#########4| 29911/31509 [00:51<00:02, 680.61it/s]","\rparsing log, completed traces ::  95%|#########5| 29980/31509 [00:51<00:02, 681.81it/s]","\rparsing log, completed traces ::  95%|#########5| 30049/31509 [00:51<00:02, 669.96it/s]","\rparsing log, completed traces ::  96%|#########5| 30122/31509 [00:51<00:02, 684.32it/s]","\rparsing log, completed traces ::  96%|#########5| 30191/31509 [00:51<00:01, 682.65it/s]","\rparsing log, completed traces ::  96%|#########6| 30263/31509 [00:51<00:01, 693.49it/s]","\rparsing log, completed traces ::  96%|#########6| 30335/31509 [00:51<00:01, 700.84it/s]","\rparsing log, completed traces ::  97%|#########6| 30408/31509 [00:51<00:01, 707.10it/s]","\rparsing log, completed traces ::  97%|#########6| 30479/31509 [00:51<00:01, 698.98it/s]","\rparsing log, completed traces ::  97%|#########6| 30549/31509 [00:52<00:01, 698.26it/s]","\rparsing log, completed traces ::  97%|#########7| 30619/31509 [00:53<00:05, 167.24it/s]","\rparsing log, completed traces ::  97%|#########7| 30690/31509 [00:53<00:03, 217.07it/s]","\rparsing log, completed traces ::  98%|#########7| 30759/31509 [00:53<00:02, 271.78it/s]","\rparsing log, completed traces ::  98%|#########7| 30832/31509 [00:53<00:02, 336.83it/s]","\rparsing log, completed traces ::  98%|#########8| 30900/31509 [00:53<00:01, 394.75it/s]","\rparsing log, completed traces ::  98%|#########8| 30972/31509 [00:53<00:01, 457.22it/s]","\rparsing log, completed traces ::  99%|#########8| 31042/31509 [00:53<00:00, 509.66it/s]","\rparsing log, completed traces ::  99%|#########8| 31119/31509 [00:53<00:00, 570.59it/s]","\rparsing log, completed traces ::  99%|#########8| 31192/31509 [00:54<00:00, 609.93it/s]","\rparsing log, completed traces ::  99%|#########9| 31264/31509 [00:54<00:00, 617.98it/s]","\rparsing log, completed traces ::  99%|#########9| 31333/31509 [00:54<00:00, 622.42it/s]","\rparsing log, completed traces :: 100%|#########9| 31401/31509 [00:54<00:00, 636.98it/s]","\rparsing log, completed traces :: 100%|#########9| 31471/31509 [00:54<00:00, 654.46it/s]","","\rparsing log, completed traces :: 100%|##########| 31509/31509 [00:54<00:00, 577.61it/s]","\n","[data] Loading XES: /workspace/data/Road_Traffic_Fine_Management_Process.xes","\n","\rparsing log, completed traces ::   0%|          | 0/150370 [00:00<?, ?it/s]","\rparsing log, completed traces ::   1%|          | 774/150370 [00:00<00:19, 7738.75it/s]","\rparsing log, completed traces ::   1%|1         | 1601/150370 [00:00<00:18, 8019.37it/s]","\rparsing log, completed traces ::   2%|1         | 2432/150370 [00:00<00:18, 8150.10it/s]","\rparsing log, completed traces ::   2%|2         | 3247/150370 [00:00<00:18, 8114.20it/s]","\rparsing log, completed traces ::   3%|2         | 4059/150370 [00:00<00:18, 8103.72it/s]","\rparsing log, completed traces ::   3%|3         | 4872/150370 [00:00<00:17, 8090.17it/s]","\rparsing log, completed traces ::   4%|3         | 5694/150370 [00:00<00:17, 8132.01it/s]","\rparsing log, completed traces ::   4%|4         | 6508/150370 [00:00<00:17, 8111.89it/s]","\rparsing log, completed traces ::   5%|4         | 7321/150370 [00:00<00:17, 8094.11it/s]","\rparsing log, completed traces ::   5%|5         | 8176/150370 [00:01<00:17, 8233.96it/s]","\rparsing log, completed traces ::   6%|6         | 9047/150370 [00:01<00:16, 8365.30it/s]","\rparsing log, completed traces ::   7%|6         | 9889/150370 [00:01<00:16, 8381.73it/s]","\rparsing log, completed traces ::   7%|7         | 10728/150370 [00:01<00:31, 4399.88it/s]","\rparsing log, completed traces ::   8%|7         | 11550/150370 [00:01<00:27, 5105.00it/s]","\rparsing log, completed traces ::   8%|8         | 12355/150370 [00:01<00:24, 5719.34it/s]","\rparsing log, completed traces ::   9%|8         | 13154/150370 [00:01<00:21, 6240.75it/s]","\rparsing log, completed traces ::   9%|9         | 13933/150370 [00:02<00:20, 6617.60it/s]","\rparsing log, completed traces ::  10%|9         | 14739/150370 [00:02<00:19, 6992.73it/s]","\rparsing log, completed traces ::  10%|#         | 15581/150370 [00:02<00:18, 7369.18it/s]","\rparsing log, completed traces ::  11%|#         | 16439/150370 [00:02<00:17, 7704.79it/s]","\rparsing log, completed traces ::  12%|#1        | 17311/150370 [00:02<00:16, 7992.93it/s]","\rparsing log, completed traces ::  12%|#2        | 18158/150370 [00:02<00:16, 8129.62it/s]","\rparsing log, completed traces ::  13%|#2        | 19005/150370 [00:02<00:16, 8205.14it/s]","\rparsing log, completed traces ::  13%|#3        | 19847/150370 [00:02<00:15, 8267.44it/s]","\rparsing log, completed traces ::  14%|#3        | 20687/150370 [00:02<00:15, 8286.01it/s]","\rparsing log, completed traces ::  14%|#4        | 21595/150370 [00:02<00:15, 8515.43it/s]","\rparsing log, completed traces ::  15%|#4        | 22469/150370 [00:03<00:14, 8581.90it/s]","\rparsing log, completed traces ::  16%|#5        | 23347/150370 [00:03<00:14, 8640.89it/s]","\rparsing log, completed traces ::  16%|#6        | 24214/150370 [00:03<00:30, 4161.01it/s]","\rparsing log, completed traces ::  17%|#6        | 24900/150370 [00:03<00:27, 4618.04it/s]","\rparsing log, completed traces ::  17%|#7        | 25617/150370 [00:03<00:24, 5108.31it/s]","\rparsing log, completed traces ::  17%|#7        | 26299/150370 [00:03<00:22, 5460.18it/s]","\rparsing log, completed traces ::  18%|#7        | 26978/150370 [00:03<00:21, 5738.68it/s]","\rparsing log, completed traces ::  18%|#8        | 27652/150370 [00:04<00:20, 5966.91it/s]","\rparsing log, completed traces ::  19%|#8        | 28378/150370 [00:04<00:19, 6306.33it/s]","\rparsing log, completed traces ::  19%|#9        | 29183/150370 [00:04<00:17, 6782.30it/s]","\rparsing log, completed traces ::  20%|#9        | 29946/150370 [00:04<00:17, 7019.99it/s]","\rparsing log, completed traces ::  20%|##        | 30681/150370 [00:04<00:17, 7007.87it/s]","\rparsing log, completed traces ::  21%|##        | 31405/150370 [00:04<00:17, 6973.23it/s]","\rparsing log, completed traces ::  21%|##1       | 32119/150370 [00:04<00:16, 6959.86it/s]","\rparsing log, completed traces ::  22%|##1       | 32827/150370 [00:04<00:16, 6933.90it/s]","\rparsing log, completed traces ::  22%|##2       | 33535/150370 [00:04<00:16, 6975.64it/s]","\rparsing log, completed traces ::  23%|##2       | 34250/150370 [00:04<00:16, 7003.28it/s]","\rparsing log, completed traces ::  23%|##3       | 34955/150370 [00:05<00:16, 6988.77it/s]","\rparsing log, completed traces ::  24%|##3       | 35657/150370 [00:05<00:16, 6992.01it/s]","\rparsing log, completed traces ::  24%|##4       | 36359/150370 [00:05<00:35, 3214.62it/s]","\rparsing log, completed traces ::  25%|##4       | 37078/150370 [00:05<00:29, 3861.93it/s]","\rparsing log, completed traces ::  25%|##5       | 37788/150370 [00:05<00:25, 4472.53it/s]","\rparsing log, completed traces ::  26%|##5       | 38531/150370 [00:05<00:21, 5099.71it/s]","\rparsing log, completed traces ::  26%|##6       | 39278/150370 [00:06<00:19, 5650.74it/s]","\rparsing log, completed traces ::  27%|##6       | 40029/150370 [00:06<00:18, 6114.57it/s]","\rparsing log, completed traces ::  27%|##7       | 40778/150370 [00:06<00:16, 6466.69it/s]","\rparsing log, completed traces ::  28%|##7       | 41562/150370 [00:06<00:15, 6827.94it/s]","\rparsing log, completed traces ::  28%|##8       | 42350/150370 [00:06<00:15, 7114.17it/s]","\rparsing log, completed traces ::  29%|##8       | 43124/150370 [00:06<00:14, 7288.73it/s]","\rparsing log, completed traces ::  29%|##9       | 43895/150370 [00:06<00:14, 7406.40it/s]","\rparsing log, completed traces ::  30%|##9       | 44659/150370 [00:06<00:14, 7472.28it/s]","\rparsing log, completed traces ::  30%|###       | 45421/150370 [00:06<00:14, 7455.18it/s]","\rparsing log, completed traces ::  31%|###       | 46177/150370 [00:06<00:13, 7459.51it/s]","\rparsing log, completed traces ::  31%|###1      | 46930/150370 [00:07<00:13, 7416.27it/s]","\rparsing log, completed traces ::  32%|###1      | 47677/150370 [00:07<00:13, 7389.22it/s]","\rparsing log, completed traces ::  32%|###2      | 48469/150370 [00:07<00:13, 7545.83it/s]","\rparsing log, completed traces ::  33%|###2      | 49338/150370 [00:07<00:12, 7878.84it/s]","\rparsing log, completed traces ::  33%|###3      | 50129/150370 [00:07<00:12, 7829.75it/s]","\rparsing log, completed traces ::  34%|###3      | 50914/150370 [00:07<00:12, 7736.35it/s]","\rparsing log, completed traces ::  34%|###4      | 51689/150370 [00:08<00:30, 3281.20it/s]","\rparsing log, completed traces ::  35%|###4      | 52441/150370 [00:08<00:24, 3923.47it/s]","\rparsing log, completed traces ::  35%|###5      | 53202/150370 [00:08<00:21, 4579.24it/s]","\rparsing log, completed traces ::  36%|###5      | 53969/150370 [00:08<00:18, 5204.93it/s]","\rparsing log, completed traces ::  36%|###6      | 54793/150370 [00:08<00:16, 5886.37it/s]","\rparsing log, completed traces ::  37%|###6      | 55576/150370 [00:08<00:14, 6347.92it/s]","\rparsing log, completed traces ::  38%|###7      | 56392/150370 [00:08<00:13, 6813.59it/s]","\rparsing log, completed traces ::  38%|###8      | 57206/150370 [00:08<00:13, 7152.24it/s]","\rparsing log, completed traces ::  39%|###8      | 57996/150370 [00:08<00:12, 7343.16it/s]","\rparsing log, completed traces ::  39%|###9      | 58778/150370 [00:09<00:12, 7424.18it/s]","\rparsing log, completed traces ::  40%|###9      | 59587/150370 [00:09<00:11, 7601.49it/s]","\rparsing log, completed traces ::  40%|####      | 60386/150370 [00:09<00:11, 7708.22it/s]","\rparsing log, completed traces ::  41%|####      | 61263/150370 [00:09<00:11, 8017.84it/s]","\rparsing log, completed traces ::  41%|####1     | 62162/150370 [00:09<00:10, 8280.44it/s]","\rparsing log, completed traces ::  42%|####1     | 63000/150370 [00:09<00:10, 8203.84it/s]","\rparsing log, completed traces ::  42%|####2     | 63827/150370 [00:09<00:10, 8118.01it/s]","\rparsing log, completed traces ::  43%|####2     | 64653/150370 [00:09<00:10, 8134.46it/s]","\rparsing log, completed traces ::  44%|####3     | 65470/150370 [00:09<00:10, 8124.48it/s]","\rparsing log, completed traces ::  44%|####4     | 66324/150370 [00:09<00:10, 8222.43it/s]","\rparsing log, completed traces ::  45%|####4     | 67148/150370 [00:10<00:10, 8218.23it/s]","\rparsing log, completed traces ::  45%|####5     | 68057/150370 [00:10<00:09, 8476.38it/s]","\rparsing log, completed traces ::  46%|####5     | 68912/150370 [00:10<00:09, 8496.42it/s]","\rparsing log, completed traces ::  46%|####6     | 69763/150370 [00:10<00:09, 8379.32it/s]","\rparsing log, completed traces ::  47%|####6     | 70602/150370 [00:10<00:09, 8260.83it/s]","\rparsing log, completed traces ::  48%|####7     | 71429/150370 [00:11<00:24, 3258.76it/s]","\rparsing log, completed traces ::  48%|####8     | 72229/150370 [00:11<00:19, 3932.56it/s]","\rparsing log, completed traces ::  49%|####8     | 72995/150370 [00:11<00:16, 4561.48it/s]","\rparsing log, completed traces ::  49%|####9     | 73740/150370 [00:11<00:14, 5122.56it/s]","\rparsing log, completed traces ::  50%|####9     | 74630/150370 [00:11<00:12, 5942.66it/s]","\rparsing log, completed traces ::  50%|#####     | 75470/150370 [00:11<00:11, 6522.22it/s]","\rparsing log, completed traces ::  51%|#####     | 76321/150370 [00:11<00:10, 7013.03it/s]","\rparsing log, completed traces ::  51%|#####1    | 77127/150370 [00:11<00:10, 7271.09it/s]","\rparsing log, completed traces ::  52%|#####1    | 77931/150370 [00:11<00:09, 7257.10it/s]","\rparsing log, completed traces ::  52%|#####2    | 78710/150370 [00:12<00:09, 7328.58it/s]","\rparsing log, completed traces ::  53%|#####2    | 79481/150370 [00:12<00:09, 7418.06it/s]","\rparsing log, completed traces ::  53%|#####3    | 80252/150370 [00:12<00:09, 7501.25it/s]","\rparsing log, completed traces ::  54%|#####3    | 81025/150370 [00:12<00:09, 7565.76it/s]","\rparsing log, completed traces ::  54%|#####4    | 81797/150370 [00:12<00:09, 7608.85it/s]","\rparsing log, completed traces ::  55%|#####4    | 82661/150370 [00:12<00:08, 7911.52it/s]","\rparsing log, completed traces ::  56%|#####5    | 83460/150370 [00:12<00:08, 7914.23it/s]","\rparsing log, completed traces ::  56%|#####6    | 84257/150370 [00:12<00:08, 7771.80it/s]","\rparsing log, completed traces ::  57%|#####6    | 85039/150370 [00:12<00:08, 7565.67it/s]","\rparsing log, completed traces ::  57%|#####7    | 85800/150370 [00:12<00:08, 7363.15it/s]","\rparsing log, completed traces ::  58%|#####7    | 86540/150370 [00:13<00:08, 7344.00it/s]","\rparsing log, completed traces ::  58%|#####8    | 87298/150370 [00:13<00:08, 7411.42it/s]","\rparsing log, completed traces ::  59%|#####8    | 88046/150370 [00:13<00:08, 7413.45it/s]","\rparsing log, completed traces ::  59%|#####9    | 88860/150370 [00:13<00:08, 7611.40it/s]","\rparsing log, completed traces ::  60%|#####9    | 89705/150370 [00:13<00:07, 7858.46it/s]","\rparsing log, completed traces ::  60%|######    | 90528/150370 [00:13<00:07, 7966.97it/s]","\rparsing log, completed traces ::  61%|######    | 91348/150370 [00:13<00:07, 8034.30it/s]","\rparsing log, completed traces ::  61%|######1   | 92182/150370 [00:13<00:07, 8116.73it/s]","\rparsing log, completed traces ::  62%|######1   | 92995/150370 [00:14<00:19, 2913.73it/s]","\rparsing log, completed traces ::  62%|######2   | 93809/150370 [00:14<00:15, 3606.72it/s]","\rparsing log, completed traces ::  63%|######2   | 94676/150370 [00:14<00:12, 4406.65it/s]","\rparsing log, completed traces ::  64%|######3   | 95529/150370 [00:14<00:10, 5160.27it/s]","\rparsing log, completed traces ::  64%|######4   | 96357/150370 [00:14<00:09, 5811.06it/s]","\rparsing log, completed traces ::  65%|######4   | 97169/150370 [00:14<00:08, 6340.79it/s]","\rparsing log, completed traces ::  65%|######5   | 97959/150370 [00:15<00:07, 6714.42it/s]","\rparsing log, completed traces ::  66%|######5   | 98774/150370 [00:15<00:07, 7086.49it/s]","\rparsing log, completed traces ::  66%|######6   | 99570/150370 [00:15<00:06, 7310.30it/s]","\rparsing log, completed traces ::  67%|######6   | 100376/150370 [00:15<00:06, 7518.83it/s]","\rparsing log, completed traces ::  67%|######7   | 101189/150370 [00:15<00:06, 7692.82it/s]","\rparsing log, completed traces ::  68%|######7   | 102019/150370 [00:15<00:06, 7854.85it/s]","\rparsing log, completed traces ::  68%|######8   | 102834/150370 [00:15<00:05, 7941.00it/s]","\rparsing log, completed traces ::  69%|######8   | 103645/150370 [00:15<00:05, 7957.06it/s]","\rparsing log, completed traces ::  69%|######9   | 104453/150370 [00:15<00:05, 7929.69it/s]","\rparsing log, completed traces ::  70%|######9   | 105255/150370 [00:15<00:05, 7903.83it/s]","\rparsing log, completed traces ::  71%|#######   | 106052/150370 [00:16<00:05, 7892.56it/s]","\rparsing log, completed traces ::  71%|#######1  | 106849/150370 [00:16<00:05, 7915.18it/s]","\rparsing log, completed traces ::  72%|#######1  | 107644/150370 [00:16<00:05, 7896.90it/s]","\rparsing log, completed traces ::  72%|#######2  | 108436/150370 [00:16<00:05, 7845.00it/s]","\rparsing log, completed traces ::  73%|#######2  | 109223/150370 [00:16<00:05, 7842.65it/s]","\rparsing log, completed traces ::  73%|#######3  | 110019/150370 [00:16<00:05, 7876.82it/s]","\rparsing log, completed traces ::  74%|#######3  | 110808/150370 [00:16<00:05, 7851.02it/s]","\rparsing log, completed traces ::  74%|#######4  | 111595/150370 [00:16<00:04, 7833.97it/s]","\rparsing log, completed traces ::  75%|#######4  | 112379/150370 [00:16<00:04, 7779.41it/s]","\rparsing log, completed traces ::  75%|#######5  | 113158/150370 [00:16<00:04, 7665.90it/s]","\rparsing log, completed traces ::  76%|#######5  | 114024/150370 [00:17<00:04, 7957.66it/s]","\rparsing log, completed traces ::  76%|#######6  | 114991/150370 [00:17<00:04, 8463.64it/s]","\rparsing log, completed traces ::  77%|#######7  | 115939/150370 [00:17<00:03, 8765.65it/s]","\rparsing log, completed traces ::  78%|#######7  | 116934/150370 [00:17<00:03, 9118.29it/s]","\rparsing log, completed traces ::  78%|#######8  | 117847/150370 [00:17<00:03, 8952.25it/s]","\rparsing log, completed traces ::  79%|#######8  | 118744/150370 [00:18<00:11, 2866.08it/s]","\rparsing log, completed traces ::  79%|#######9  | 119463/150370 [00:18<00:09, 3379.67it/s]","\rparsing log, completed traces ::  80%|#######9  | 120195/150370 [00:18<00:07, 3951.57it/s]","\rparsing log, completed traces ::  80%|########  | 120922/150370 [00:18<00:06, 4521.43it/s]","\rparsing log, completed traces ::  81%|########  | 121662/150370 [00:18<00:05, 5088.53it/s]","\rparsing log, completed traces ::  81%|########1 | 122415/150370 [00:18<00:04, 5613.30it/s]","\rparsing log, completed traces ::  82%|########1 | 123169/150370 [00:18<00:04, 6071.02it/s]","\rparsing log, completed traces ::  82%|########2 | 123902/150370 [00:19<00:04, 6380.07it/s]","\rparsing log, completed traces ::  83%|########2 | 124634/150370 [00:19<00:03, 6598.56it/s]","\rparsing log, completed traces ::  83%|########3 | 125377/150370 [00:19<00:03, 6825.19it/s]","\rparsing log, completed traces ::  84%|########3 | 126136/150370 [00:19<00:03, 7039.88it/s]","\rparsing log, completed traces ::  84%|########4 | 126902/150370 [00:19<00:03, 7214.86it/s]","\rparsing log, completed traces ::  85%|########4 | 127653/150370 [00:19<00:03, 7298.23it/s]","\rparsing log, completed traces ::  85%|########5 | 128411/150370 [00:19<00:02, 7368.81it/s]","\rparsing log, completed traces ::  86%|########5 | 129197/150370 [00:19<00:02, 7512.46it/s]","\rparsing log, completed traces ::  86%|########6 | 129963/150370 [00:19<00:02, 7552.21it/s]","\rparsing log, completed traces ::  87%|########6 | 130725/150370 [00:19<00:02, 7567.39it/s]","\rparsing log, completed traces ::  87%|########7 | 131493/150370 [00:20<00:02, 7600.78it/s]","\rparsing log, completed traces ::  88%|########8 | 132397/150370 [00:20<00:02, 8029.31it/s]","\rparsing log, completed traces ::  89%|########8 | 133252/150370 [00:20<00:02, 8184.17it/s]","\rparsing log, completed traces ::  89%|########9 | 134073/150370 [00:20<00:02, 8078.56it/s]","\rparsing log, completed traces ::  90%|########9 | 134883/150370 [00:20<00:01, 8000.21it/s]","\rparsing log, completed traces ::  90%|######### | 135685/150370 [00:20<00:01, 7878.56it/s]","\rparsing log, completed traces ::  91%|######### | 136474/150370 [00:20<00:01, 7765.57it/s]","\rparsing log, completed traces ::  91%|#########1| 137252/150370 [00:20<00:01, 7726.37it/s]","\rparsing log, completed traces ::  92%|#########1| 138026/150370 [00:20<00:01, 7672.65it/s]","\rparsing log, completed traces ::  92%|#########2| 138794/150370 [00:20<00:01, 7669.53it/s]","\rparsing log, completed traces ::  93%|#########2| 139562/150370 [00:21<00:01, 7582.52it/s]","\rparsing log, completed traces ::  93%|#########3| 140328/150370 [00:21<00:01, 7604.92it/s]","\rparsing log, completed traces ::  94%|#########3| 141113/150370 [00:21<00:01, 7676.25it/s]","\rparsing log, completed traces ::  94%|#########4| 141912/150370 [00:21<00:01, 7769.49it/s]","\rparsing log, completed traces ::  95%|#########4| 142690/150370 [00:21<00:00, 7695.05it/s]","\rparsing log, completed traces ::  95%|#########5| 143461/150370 [00:21<00:00, 7699.32it/s]","\rparsing log, completed traces ::  96%|#########5| 144232/150370 [00:21<00:00, 7666.75it/s]","\rparsing log, completed traces ::  96%|#########6| 144999/150370 [00:21<00:00, 7537.49it/s]","\rparsing log, completed traces ::  97%|#########6| 145754/150370 [00:21<00:00, 7122.42it/s]","\rparsing log, completed traces ::  97%|#########7| 146471/150370 [00:22<00:01, 2127.09it/s]","\rparsing log, completed traces ::  98%|#########7| 147097/150370 [00:22<00:01, 2573.18it/s]","\rparsing log, completed traces ::  98%|#########8| 147705/150370 [00:23<00:00, 3039.71it/s]","\rparsing log, completed traces ::  99%|#########8| 148330/150370 [00:23<00:00, 3553.96it/s]","\rparsing log, completed traces ::  99%|#########9| 148990/150370 [00:23<00:00, 4117.66it/s]","\rparsing log, completed traces :: 100%|#########9| 149683/150370 [00:23<00:00, 4708.21it/s]","","\rparsing log, completed traces :: 100%|##########| 150370/150370 [00:23<00:00, 6422.82it/s]","\n","[data] Loaded datasets: ['BPI2012', 'BPI2017', 'ROAD']","\n","\n=== Dataset: BPI2012 ===","\n","Samples train/val/test: 22252/4176/4196; vocab=23","\n","Epoch 1: validation_loss = 0.5594 | val_acc=0.7409 | val_f1=0.5316 | val_top3=0.9840","\n","Epoch 2: validation_loss = 0.5308 | val_acc=0.7598 | val_f1=0.5425 | val_top3=0.9854","\n","Epoch 3: validation_loss = 0.5333 | val_acc=0.7574 | val_f1=0.5725 | val_top3=0.9859","\n","Epoch 4: validation_loss = 0.5195 | val_acc=0.7629 | val_f1=0.5835 | val_top3=0.9861","\n","Epoch 5: validation_loss = 0.5221 | val_acc=0.7450 | val_f1=0.6007 | val_top3=0.9854","\n","Epoch 6: validation_loss = 0.5249 | val_acc=0.7634 | val_f1=0.5833 | val_top3=0.9861","\n","Epoch 7: validation_loss = 0.5073 | val_acc=0.7639 | val_f1=0.5909 | val_top3=0.9856","\n","Epoch 8: validation_loss = 0.5113 | val_acc=0.7593 | val_f1=0.5790 | val_top3=0.9859","\n","Epoch 9: validation_loss = 0.5105 | val_acc=0.7639 | val_f1=0.5842 | val_top3=0.9852","\n","Epoch 10: validation_loss = 0.5201 | val_acc=0.7634 | val_f1=0.5797 | val_top3=0.9847","\n","[BPI2012] Train: loss=0.5148 acc=0.7777 f1=0.5609 top3=0.9868","\n","[BPI2012] Test:  loss=0.5355 acc=0.7569 f1=0.5872 top3=0.9874","\n","\n=== Dataset: BPI2017 ===","\n","Samples train/val/test: 34519/7403/7374; vocab=24","\n","Epoch 1: validation_loss = 0.4172 | val_acc=0.8368 | val_f1=0.5299 | val_top3=0.9904","\n","Epoch 2: validation_loss = 0.4004 | val_acc=0.8367 | val_f1=0.5595 | val_top3=0.9907","\n","Epoch 3: validation_loss = 0.3864 | val_acc=0.8384 | val_f1=0.5719 | val_top3=0.9912","\n","Epoch 4: validation_loss = 0.3847 | val_acc=0.8395 | val_f1=0.5903 | val_top3=0.9919","\n","Epoch 5: validation_loss = 0.3812 | val_acc=0.8405 | val_f1=0.5856 | val_top3=0.9922","\n","Epoch 6: validation_loss = 0.3802 | val_acc=0.8376 | val_f1=0.5896 | val_top3=0.9916","\n","Epoch 7: validation_loss = 0.3769 | val_acc=0.8361 | val_f1=0.5957 | val_top3=0.9924","\n","Epoch 8: validation_loss = 0.3799 | val_acc=0.8386 | val_f1=0.6180 | val_top3=0.9927","\n","Epoch 9: validation_loss = 0.3805 | val_acc=0.8391 | val_f1=0.5989 | val_top3=0.9926","\n","Epoch 10: validation_loss = 0.3756 | val_acc=0.8399 | val_f1=0.6155 | val_top3=0.9928","\n","[BPI2017] Train: loss=0.3607 acc=0.8422 f1=0.5721 top3=0.9941","\n","[BPI2017] Test:  loss=0.3877 acc=0.8332 f1=0.5710 top3=0.9906","\n","\n=== Dataset: ROAD ===","\n","Samples train/val/test: 8707/1928/1869; vocab=11","\n","Epoch 1: validation_loss = 0.4913 | val_acc=0.7993 | val_f1=0.3684 | val_top3=0.9922","\n","Epoch 2: validation_loss = 0.4561 | val_acc=0.8086 | val_f1=0.4514 | val_top3=0.9948","\n","Epoch 3: validation_loss = 0.4566 | val_acc=0.8091 | val_f1=0.5300 | val_top3=0.9969","\n","Epoch 4: validation_loss = 0.4482 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 5: validation_loss = 0.4355 | val_acc=0.8112 | val_f1=0.6427 | val_top3=0.9969","\n","Epoch 6: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 7: validation_loss = 0.4328 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 8: validation_loss = 0.4334 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 9: validation_loss = 0.4274 | val_acc=0.8117 | val_f1=0.6438 | val_top3=0.9969","\n","Epoch 10: validation_loss = 0.4329 | val_acc=0.8122 | val_f1=0.6664 | val_top3=0.9969","\n","[ROAD] Train: loss=0.4662 acc=0.7894 f1=0.5395 top3=0.9986","\n","[ROAD] Test:  loss=0.4833 acc=0.8020 f1=0.4740 top3=0.9936","\n","Execution time: 3 minutes seconds (time limit is an hour)."],"parse_metrics_plan":"I will load the numpy file from the specified working directory, parse the nested experiment_data structure, and compute final or best metrics as requested. For training and test, I will report the final metrics saved in the structure. For validation, I will compute the best values across epochs for accuracy, F1 score, and top-3 accuracy, and the minimum validation loss using the recorded loss history. The script prints the dataset name followed by clearly labeled metrics and executes immediately.","parse_metrics_code":"import os\nimport numpy as np\n\n\ndef safe_get(d, *keys, default=None):\n    cur = d\n    for k in keys:\n        if isinstance(cur, dict) and k in cur:\n            cur = cur[k]\n        else:\n            return default\n    return cur\n\n\ndef load_experiment_data():\n    working_dir = os.path.join(os.getcwd(), \"working\")\n    path = os.path.join(working_dir, \"experiment_data.npy\")\n    if not os.path.isfile(path):\n        raise FileNotFoundError(f\"File not found: {path}\")\n    data = np.load(path, allow_pickle=True).item()\n    if not isinstance(data, dict):\n        raise ValueError(\"Loaded experiment_data is not a dict.\")\n    return data\n\n\ndef extract_final_train_metrics(ds_data):\n    # Expect a 'final' entry\n    train_entries = safe_get(ds_data, \"metrics\", \"train\", default=[])\n    final = None\n    for tag, vals in train_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"train loss\"] = final.get(\"loss\", None)\n        out[\"train accuracy\"] = final.get(\"acc\", None)\n        out[\"train F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"train top-3 accuracy\"] = final.get(\"top3\", None)\n    else:\n        # Fallback: try last value from losses for loss only\n        train_losses = safe_get(ds_data, \"losses\", \"train\", default=[])\n        if train_losses:\n            out[\"train loss\"] = train_losses[-1][1]\n    return out\n\n\ndef extract_best_val_metrics(ds_data):\n    # Collect per-epoch validation metrics (acc, macro_f1, top3)\n    val_entries = safe_get(ds_data, \"metrics\", \"val\", default=[])\n    best = {\n        \"validation accuracy\": None,\n        \"validation F1 score\": None,\n        \"validation top-3 accuracy\": None,\n    }\n    for tag, vals in val_entries:\n        if isinstance(vals, dict):\n            acc = vals.get(\"acc\", None)\n            f1 = vals.get(\"macro_f1\", None)\n            top3 = vals.get(\"top3\", None)\n            if acc is not None:\n                best[\"validation accuracy\"] = (\n                    acc\n                    if best[\"validation accuracy\"] is None\n                    else max(best[\"validation accuracy\"], acc)\n                )\n            if f1 is not None:\n                best[\"validation F1 score\"] = (\n                    f1\n                    if best[\"validation F1 score\"] is None\n                    else max(best[\"validation F1 score\"], f1)\n                )\n            if top3 is not None:\n                best[\"validation top-3 accuracy\"] = (\n                    top3\n                    if best[\"validation top-3 accuracy\"] is None\n                    else max(best[\"validation top-3 accuracy\"], top3)\n                )\n    # For validation loss, use recorded losses per epoch and take the minimum\n    val_losses = safe_get(ds_data, \"losses\", \"val\", default=[])\n    if val_losses:\n        min_val_loss = min((v for (_, v) in val_losses), default=None)\n    else:\n        # fallback: check 'final' val entry if present\n        min_val_loss = None\n        for tag, vals in val_entries:\n            if tag == \"final\" and isinstance(vals, dict):\n                min_val_loss = vals.get(\"loss\", None)\n                break\n    out = {\"validation loss\": min_val_loss}\n    out.update(best)\n    return out\n\n\ndef extract_final_test_metrics(ds_data):\n    test_entries = safe_get(ds_data, \"metrics\", \"test\", default=[])\n    final = None\n    for tag, vals in test_entries:\n        if tag == \"final\":\n            final = vals\n    out = {}\n    if final:\n        out[\"test loss\"] = final.get(\"loss\", None)\n        out[\"test accuracy\"] = final.get(\"acc\", None)\n        out[\"test F1 score\"] = final.get(\"macro_f1\", None)\n        out[\"test top-3 accuracy\"] = final.get(\"top3\", None)\n    return out\n\n\ndef format_print_metrics(dataset_name, metrics_dict):\n    print(dataset_name)\n    # Keep a consistent order for readability\n    ordered_keys = [\n        \"train loss\",\n        \"train accuracy\",\n        \"train F1 score\",\n        \"train top-3 accuracy\",\n        \"validation loss\",\n        \"validation accuracy\",\n        \"validation F1 score\",\n        \"validation top-3 accuracy\",\n        \"test loss\",\n        \"test accuracy\",\n        \"test F1 score\",\n        \"test top-3 accuracy\",\n    ]\n    printed_any = False\n    for k in ordered_keys:\n        if k in metrics_dict and metrics_dict[k] is not None:\n            val = metrics_dict[k]\n            if isinstance(val, float):\n                print(f\"{k}: {val:.4f}\")\n            else:\n                print(f\"{k}: {val}\")\n            printed_any = True\n    # Print any remaining keys that were not in the ordered list\n    for k, v in metrics_dict.items():\n        if k not in ordered_keys and v is not None:\n            if isinstance(v, float):\n                print(f\"{k}: {v:.4f}\")\n            else:\n                print(f\"{k}: {v}\")\n            printed_any = True\n    if not printed_any:\n        print(\"No metrics available\")\n\n\ndef run():\n    experiment_data = load_experiment_data()\n    for ds_name, ds_data in experiment_data.items():\n        all_metrics = {}\n        all_metrics.update(extract_final_train_metrics(ds_data))\n        all_metrics.update(extract_best_val_metrics(ds_data))\n        all_metrics.update(extract_final_test_metrics(ds_data))\n        format_print_metrics(ds_name, all_metrics)\n\n\n# Execute immediately\nrun()\n","parse_term_out":["BPI2012","\n","train loss: 0.5148","\n","train accuracy: 0.7777","\n","train F1 score: 0.5609","\n","train top-3 accuracy: 0.9868","\n","validation loss: 0.5073","\n","validation accuracy: 0.7639","\n","validation F1 score: 0.6007","\n","validation top-3 accuracy: 0.9861","\n","test loss: 0.5355","\n","test accuracy: 0.7569","\n","test F1 score: 0.5872","\n","test top-3 accuracy: 0.9874","\n","BPI2017","\n","train loss: 0.3607","\n","train accuracy: 0.8422","\n","train F1 score: 0.5721","\n","train top-3 accuracy: 0.9941","\n","validation loss: 0.3756","\n","validation accuracy: 0.8405","\n","validation F1 score: 0.6180","\n","validation top-3 accuracy: 0.9928","\n","test loss: 0.3877","\n","test accuracy: 0.8332","\n","test F1 score: 0.5710","\n","test top-3 accuracy: 0.9906","\n","ROAD","\n","train loss: 0.4662","\n","train accuracy: 0.7894","\n","train F1 score: 0.5395","\n","train top-3 accuracy: 0.9986","\n","validation loss: 0.4274","\n","validation accuracy: 0.8122","\n","validation F1 score: 0.6664","\n","validation top-3 accuracy: 0.9969","\n","test loss: 0.4833","\n","test accuracy: 0.8020","\n","test F1 score: 0.4740","\n","test top-3 accuracy: 0.9936","\n","Execution time: a moment seconds (time limit is an hour)."],"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":235.2198452949524,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":"","exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088","metric":{"value":{"metric_names":[{"metric_name":"train loss","lower_is_better":true,"description":"Cross-entropy loss on the training split","data":[{"dataset_name":"BPI2012","final_value":0.5148,"best_value":0.5148},{"dataset_name":"BPI2017","final_value":0.3607,"best_value":0.3607},{"dataset_name":"ROAD","final_value":0.4662,"best_value":0.4662}]},{"metric_name":"train accuracy","lower_is_better":false,"description":"Classification accuracy on the training split","data":[{"dataset_name":"BPI2012","final_value":0.7777,"best_value":0.7777},{"dataset_name":"BPI2017","final_value":0.8422,"best_value":0.8422},{"dataset_name":"ROAD","final_value":0.7894,"best_value":0.7894}]},{"metric_name":"train F1 score","lower_is_better":false,"description":"F1 score on the training split","data":[{"dataset_name":"BPI2012","final_value":0.5609,"best_value":0.5609},{"dataset_name":"BPI2017","final_value":0.5721,"best_value":0.5721},{"dataset_name":"ROAD","final_value":0.5395,"best_value":0.5395}]},{"metric_name":"train top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the training split","data":[{"dataset_name":"BPI2012","final_value":0.9868,"best_value":0.9868},{"dataset_name":"BPI2017","final_value":0.9941,"best_value":0.9941},{"dataset_name":"ROAD","final_value":0.9986,"best_value":0.9986}]},{"metric_name":"validation loss","lower_is_better":true,"description":"Cross-entropy loss on the validation split","data":[{"dataset_name":"BPI2012","final_value":0.5073,"best_value":0.5073},{"dataset_name":"BPI2017","final_value":0.3756,"best_value":0.3756},{"dataset_name":"ROAD","final_value":0.4274,"best_value":0.4274}]},{"metric_name":"validation accuracy","lower_is_better":false,"description":"Classification accuracy on the validation split","data":[{"dataset_name":"BPI2012","final_value":0.7639,"best_value":0.7639},{"dataset_name":"BPI2017","final_value":0.8405,"best_value":0.8405},{"dataset_name":"ROAD","final_value":0.8122,"best_value":0.8122}]},{"metric_name":"validation F1 score","lower_is_better":false,"description":"F1 score on the validation split","data":[{"dataset_name":"BPI2012","final_value":0.6007,"best_value":0.6007},{"dataset_name":"BPI2017","final_value":0.618,"best_value":0.618},{"dataset_name":"ROAD","final_value":0.6664,"best_value":0.6664}]},{"metric_name":"validation top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the validation split","data":[{"dataset_name":"BPI2012","final_value":0.9861,"best_value":0.9861},{"dataset_name":"BPI2017","final_value":0.9928,"best_value":0.9928},{"dataset_name":"ROAD","final_value":0.9969,"best_value":0.9969}]},{"metric_name":"test loss","lower_is_better":true,"description":"Cross-entropy loss on the test split","data":[{"dataset_name":"BPI2012","final_value":0.5355,"best_value":0.5355},{"dataset_name":"BPI2017","final_value":0.3877,"best_value":0.3877},{"dataset_name":"ROAD","final_value":0.4833,"best_value":0.4833}]},{"metric_name":"test accuracy","lower_is_better":false,"description":"Classification accuracy on the test split","data":[{"dataset_name":"BPI2012","final_value":0.7569,"best_value":0.7569},{"dataset_name":"BPI2017","final_value":0.8332,"best_value":0.8332},{"dataset_name":"ROAD","final_value":0.802,"best_value":0.802}]},{"metric_name":"test F1 score","lower_is_better":false,"description":"F1 score on the test split","data":[{"dataset_name":"BPI2012","final_value":0.5872,"best_value":0.5872},{"dataset_name":"BPI2017","final_value":0.571,"best_value":0.571},{"dataset_name":"ROAD","final_value":0.474,"best_value":0.474}]},{"metric_name":"test top-3 accuracy","lower_is_better":false,"description":"Top-3 accuracy on the test split","data":[{"dataset_name":"BPI2012","final_value":0.9874,"best_value":0.9874},{"dataset_name":"BPI2017","final_value":0.9906,"best_value":0.9906},{"dataset_name":"ROAD","final_value":0.9936,"best_value":0.9936}]}]},"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":false,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":["../../logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_BPI2017.png","../../logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_BPI2012.png","../../logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_ROAD.png","../../logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_BPI2017.png","../../logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_ROAD.png","../../logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_BPI2012.png"],"plot_paths":["experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_BPI2012.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_ROAD.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_BPI2017.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_ROAD.png","experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_BPI2012.png"],"plot_analyses":[{"analysis":"The plot represents the validation top-3 accuracy for the BPI2017 dataset across multiple epochs. There is a clear upward trend in top-3 accuracy as the epochs increase, indicating that the model's ability to predict one of the top-three next activities improves with training. The gradual improvement suggests that the learning process is effective for this dataset.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_BPI2017.png"},{"analysis":"This plot shows the validation top-3 accuracy for the BPI2012 dataset. The accuracy peaks early and then fluctuates slightly, suggesting that the model quickly learns the patterns in the data but might struggle with overfitting or data peculiarities as training progresses. Overall, the accuracy remains high, indicating a generally successful prediction.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_BPI2012.png"},{"analysis":"The loss curves for the ROAD dataset depict a steep decline in both training and validation loss initially, indicating rapid learning. As the epochs progress, both losses stabilize, which suggests that the model is converging well. The validation loss slightly lower than the training loss indicates that the model generalizes well on unseen data.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_ROAD.png"},{"analysis":"This plot shows the loss curves for the BPI2017 dataset. The rapid decrease in both training and validation loss suggests effective learning. The close proximity of the two curves implies consistent performance on both training and validation sets, which is a good sign of model stability.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_BPI2017.png"},{"analysis":"The validation top-3 accuracy for the ROAD dataset starts high and quickly stabilizes, indicating strong initial performance which persists across epochs. This suggests that the model is well-suited to the characteristics of this dataset and achieves reliable predictions early in the training process.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/val_top3_ROAD.png"},{"analysis":"The loss curves for the BPI2012 dataset show a significant drop in both training and validation loss early on, followed by stabilization. The closeness of the curves indicates good generalization from training to validation data, with consistent performance across epochs.","plot_path":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/loss_curves_BPI2012.png"}],"vlm_feedback_summary":"The plots indicate successful training and prediction across different datasets with the implemented models. The high validation accuracies and consistent loss reductions across epochs suggest robust learning and effective generalization, aligning with the research hypothesis. The models show strong predictive capabilities for next-events and suffix predictions, with resource-centric approaches appearing promising.","datasets_successfully_tested":["BPI2017","BPI2012","ROAD"],"ablation_name":null,"hyperparam_name":null,"is_seed_node":true,"is_seed_agg_node":false,"exec_time_feedback":""},{"code":"# plotting aggregation code","plan":"Aggregate results from multiple seeds","overall_plan":"","plot_code":"import matplotlib.pyplot as plt\nimport numpy as np\nimport os\n\nworking_dir = os.path.join(os.getcwd(), \"working\")\nos.makedirs(working_dir, exist_ok=True)\n\n\n# Aggregate results across multiple experiment_data.npy files and plot means with SEM\ndef safe_get(d, keys, default=None):\n    cur = d\n    for k in keys:\n        if isinstance(cur, dict) and k in cur:\n            cur = cur[k]\n        else:\n            return default\n    return cur\n\n\ndef compute_sem(arr, axis=0):\n    arr = np.asarray(arr)\n    if arr.size == 0:\n        return arr\n    n = arr.shape[axis]\n    if n <= 1:\n        return np.zeros_like(np.mean(arr, axis=axis))\n    return np.std(arr, axis=axis, ddof=1) / np.sqrt(n)\n\n\ndef main():\n    # Load multiple experiment_data dicts\n    try:\n        experiment_data_path_list = [\n            \"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_9a2ff2eda58e4ea8966e9f8f6f6ba8c5_proc_332087/experiment_data.npy\",\n            \"None/experiment_data.npy\",\n            \"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/experiment_74c75ff464094554b5c0e6b72036923c_proc_332088/experiment_data.npy\",\n        ]\n        all_experiment_data = []\n        root = os.getenv(\"AI_SCIENTIST_ROOT\", \"\")\n        for rel_path in experiment_data_path_list:\n            try:\n                full_path = os.path.join(root, rel_path) if root else rel_path\n                experiment_data = np.load(full_path, allow_pickle=True).item()\n                if isinstance(experiment_data, dict):\n                    all_experiment_data.append(experiment_data)\n                    print(f\"Loaded experiment data from: {full_path}\")\n                else:\n                    print(f\"Unexpected data format in: {full_path}\")\n            except Exception as e:\n                print(f\"Error loading experiment data: {e}\")\n    except Exception as e:\n        print(f\"Error loading experiment data: {e}\")\n        all_experiment_data = []\n\n    # Merge by dataset name across runs\n    datasets = {}\n    for run_idx, exp in enumerate(all_experiment_data):\n        for ds_name, ed in exp.items():\n            if ds_name not in datasets:\n                datasets[ds_name] = {\n                    \"train_losses\": [],\n                    \"val_losses\": [],\n                    \"epochs\": [],\n                    \"prefix_lens\": [],\n                    \"top3_flags\": [],\n                    \"test_metrics\": [],  # dicts with loss, acc, macro_f1, top3\n                }\n            # losses\n            tr = ed.get(\"losses\", {}).get(\"train\", [])\n            va = ed.get(\"losses\", {}).get(\"val\", [])\n            # Extract only the y values preserving order by epoch index\n            tr_vals = [y for (_, y) in tr] if len(tr) > 0 else []\n            va_vals = [y for (_, y) in va] if len(va) > 0 else []\n            epochs = ed.get(\n                \"epochs\", list(range(1, min(len(tr_vals), len(va_vals)) + 1))\n            )\n            # store if any present\n            if len(tr_vals) > 0:\n                datasets[ds_name][\"train_losses\"].append(np.array(tr_vals, dtype=float))\n            if len(va_vals) > 0:\n                datasets[ds_name][\"val_losses\"].append(np.array(va_vals, dtype=float))\n            if len(epochs) > 0:\n                datasets[ds_name][\"epochs\"].append(np.array(epochs, dtype=int))\n\n            # top3 vs prefix length\n            pref = ed.get(\"prefix_lens\", [])\n            flags = ed.get(\"top3_flags\", [])\n            if len(pref) > 0 and len(flags) > 0:\n                datasets[ds_name][\"prefix_lens\"].append(np.array(pref, dtype=int))\n                datasets[ds_name][\"top3_flags\"].append(np.array(flags, dtype=int))\n\n            # test metrics\n            test_m = ed.get(\"metrics\", {}).get(\"test\", [])\n            if (\n                isinstance(test_m, list)\n                and len(test_m) > 0\n                and isinstance(test_m[0], (list, tuple))\n                and isinstance(test_m[0][1], dict)\n            ):\n                datasets[ds_name][\"test_metrics\"].append(dict(test_m[0][1]))\n\n    # For each dataset, create aggregated plots\n    for name, bundle in datasets.items():\n        # Aggregated loss curves (mean \u00b1 SEM) using min common length across runs\n        try:\n            tr_list = bundle[\"train_losses\"]\n            va_list = bundle[\"val_losses\"]\n\n            # Determine aligned lengths\n            if len(tr_list) > 0:\n                min_tr_len = min(len(x) for x in tr_list)\n                tr_aligned = (\n                    np.stack([x[:min_tr_len] for x in tr_list], axis=0)\n                    if min_tr_len > 0\n                    else None\n                )\n            else:\n                tr_aligned = None\n            if len(va_list) > 0:\n                min_va_len = min(len(x) for x in va_list)\n                va_aligned = (\n                    np.stack([x[:min_va_len] for x in va_list], axis=0)\n                    if min_va_len > 0\n                    else None\n                )\n            else:\n                va_aligned = None\n\n            if (tr_aligned is not None and tr_aligned.size > 0) or (\n                va_aligned is not None and va_aligned.size > 0\n            ):\n                plt.figure()\n                legend_handles = []\n                if tr_aligned is not None and tr_aligned.size > 0:\n                    x_tr = np.arange(1, tr_aligned.shape[1] + 1)\n                    tr_mean = np.mean(tr_aligned, axis=0)\n                    tr_sem = compute_sem(tr_aligned, axis=0)\n                    plt.plot(x_tr, tr_mean, color=\"tab:blue\", label=\"Train mean\")\n                    plt.fill_between(\n                        x_tr,\n                        tr_mean - tr_sem,\n                        tr_mean + tr_sem,\n                        color=\"tab:blue\",\n                        alpha=0.2,\n                        label=\"Train \u00b1 SEM\",\n                    )\n                if va_aligned is not None and va_aligned.size > 0:\n                    x_va = np.arange(1, va_aligned.shape[1] + 1)\n                    va_mean = np.mean(va_aligned, axis=0)\n                    va_sem = compute_sem(va_aligned, axis=0)\n                    plt.plot(x_va, va_mean, color=\"tab:orange\", label=\"Val mean\")\n                    plt.fill_between(\n                        x_va,\n                        va_mean - va_sem,\n                        va_mean + va_sem,\n                        color=\"tab:orange\",\n                        alpha=0.2,\n                        label=\"Val \u00b1 SEM\",\n                    )\n                plt.legend()\n                plt.xlabel(\"Epoch\")\n                plt.ylabel(\"Loss\")\n                plt.title(\n                    f\"Aggregated Loss Curves - {name}\\nLeft: Ground Truth, Right: Generated Samples | Dataset: {name}\"\n                )\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(working_dir, f\"{name}_aggregated_loss_curves.png\")\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating aggregated loss plot for {name}: {e}\")\n            plt.close()\n\n        # Aggregated Top-3 vs Prefix Length (mean \u00b1 SEM across runs)\n        try:\n            pref_runs = bundle[\"prefix_lens\"]\n            flag_runs = bundle[\"top3_flags\"]\n            if len(pref_runs) > 0 and len(flag_runs) > 0:\n                # For each run, compute per-length mean\n                per_len_run_means = {}\n                for pr, fr in zip(pref_runs, flag_runs):\n                    d = {}\n                    # group flags by prefix length for this run\n                    for L, f in zip(pr, fr):\n                        d.setdefault(int(L), []).append(int(f))\n                    # convert to mean per L for this run\n                    for L, lst in d.items():\n                        per_len_run_means.setdefault(L, []).append(float(np.mean(lst)))\n                # Aggregate across runs\n                xs = sorted(per_len_run_means.keys())\n                vals = [per_len_run_means[L] for L in xs]\n                means = np.array([np.mean(v) for v in vals], dtype=float)\n                sems = np.array(\n                    [compute_sem(np.array(v)) if len(v) > 1 else 0.0 for v in vals],\n                    dtype=float,\n                )\n                plt.figure()\n                plt.errorbar(\n                    xs, means, yerr=sems, fmt=\"-o\", capsize=3, label=\"Top-3 mean \u00b1 SEM\"\n                )\n                plt.xlabel(\"Prefix Length\")\n                plt.ylabel(\"Top-3 Accuracy\")\n                plt.title(\n                    f\"Aggregated Top-3 Accuracy vs Prefix Length - {name}\\nLeft: Ground Truth, Right: Generated Samples | Dataset: {name}\"\n                )\n                plt.legend()\n                plt.tight_layout()\n                plt.savefig(\n                    os.path.join(\n                        working_dir, f\"{name}_aggregated_top3_vs_prefixlen.png\"\n                    )\n                )\n                plt.close()\n        except Exception as e:\n            print(f\"Error creating aggregated Top-3 vs prefix length for {name}: {e}\")\n            plt.close()\n\n        # Aggregated final test metrics (bar-like error plot)\n        try:\n            tms = bundle[\"test_metrics\"]\n            if len(tms) > 0:\n                # Collect arrays\n                keys = [\"acc\", \"macro_f1\", \"top3\", \"loss\"]\n                data = {k: [] for k in keys}\n                for m in tms:\n                    for k in keys:\n                        if k in m:\n                            data[k].append(float(m[k]))\n                # Only include metrics that exist\n                present_keys = [k for k in keys if len(data[k]) > 0]\n                if len(present_keys) > 0:\n                    means = [np.mean(data[k]) for k in present_keys]\n                    sems = [\n                        compute_sem(np.array(data[k])) if len(data[k]) > 1 else 0.0\n                        for k in present_keys\n                    ]\n                    x = np.arange(len(present_keys))\n                    plt.figure()\n                    # simple point with error bars to avoid simulating bar heights\n                    plt.errorbar(\n                        x,\n                        means,\n                        yerr=sems,\n                        fmt=\"o\",\n                        capsize=5,\n                        linestyle=\"None\",\n                        label=\"Mean \u00b1 SEM\",\n                    )\n                    plt.xticks(x, present_keys)\n                    plt.ylabel(\"Value\")\n                    plt.title(\n                        f\"Aggregated Test Metrics - {name}\\nLeft: Ground Truth, Right: Generated Samples | Dataset: {name}\"\n                    )\n                    plt.legend()\n                    plt.tight_layout()\n                    plt.savefig(\n                        os.path.join(working_dir, f\"{name}_aggregated_test_metrics.png\")\n                    )\n                    plt.close()\n        except Exception as e:\n            print(f\"Error creating aggregated test metrics for {name}: {e}\")\n            plt.close()\n\n        # Print aggregated evaluation metrics\n        try:\n            tms = bundle[\"test_metrics\"]\n            if len(tms) > 0:\n                accs = [m[\"acc\"] for m in tms if \"acc\" in m]\n                f1s = [m[\"macro_f1\"] for m in tms if \"macro_f1\" in m]\n                top3s = [m[\"top3\"] for m in tms if \"top3\" in m]\n                losses = [m[\"loss\"] for m in tms if \"loss\" in m]\n\n                def stat_str(arr, name_):\n                    if len(arr) == 0:\n                        return f\"{name_}=N/A\"\n                    mean = np.mean(arr)\n                    sem = compute_sem(np.array(arr)) if len(arr) > 1 else 0.0\n                    return f\"{name_} mean={mean:.4f} \u00b1 {sem:.4f} (SEM), n={len(arr)}\"\n\n                print(\n                    f\"{name} | \"\n                    + \" | \".join(\n                        [\n                            stat_str(accs, \"acc\"),\n                            stat_str(f1s, \"macro_f1\"),\n                            stat_str(top3s, \"top3\"),\n                            stat_str(losses, \"loss\"),\n                        ]\n                    )\n                )\n        except Exception as e:\n            print(f\"Error printing metrics for {name}: {e}\")\n\n\nif __name__ == \"__main__\":\n    main()\n","plot_plan":null,"step":9,"id":"43abe3d0d55c4677add4681969dcb1b1","ctime":1757758681.4655616,"_term_out":null,"parse_metrics_plan":"","parse_metrics_code":"","parse_term_out":null,"parse_exc_type":null,"parse_exc_info":null,"parse_exc_stack":null,"exec_time":null,"exc_type":null,"exc_info":null,"exc_stack":null,"analysis":null,"exp_results_dir":"experiments/2025-09-13_11-32-42_resource_centric_ppm_agents_attempt_0/logs/0-run/experiment_results/seed_aggregation_43abe3d0d55c4677add4681969dcb1b1","metric":{"value":null,"maximize":null,"name":null,"description":null},"is_buggy":false,"is_buggy_plots":null,"parent_id":null,"children":[],"plot_data":{},"plots_generated":false,"plots":[],"plot_paths":[],"plot_analyses":[],"vlm_feedback_summary":[],"datasets_successfully_tested":[],"ablation_name":null,"hyperparam_name":null,"is_seed_node":true,"is_seed_agg_node":true,"exec_time_feedback":""}],"node2parent":{"726b2721d45c4800b9381c5d265fefcb":"637162c2bceb4502a8e96d90218586d2","9a2ff2eda58e4ea8966e9f8f6f6ba8c5":"726b2721d45c4800b9381c5d265fefcb","dba5727334664144994f0a47a97edc79":"726b2721d45c4800b9381c5d265fefcb","74c75ff464094554b5c0e6b72036923c":"726b2721d45c4800b9381c5d265fefcb","43abe3d0d55c4677add4681969dcb1b1":"726b2721d45c4800b9381c5d265fefcb"},"__version":"2"}